iOS學習筆記38-MJExtension使用
一、MJExtension第三方架構
我們在iOS開發過程中,我們常常需要將字典資料(也就是JSON資料)與Model模型之間的轉化,例如網路請求返回的微博資料、等等,如果我們自己全部手動去建立模型並賦值,都是一些毫無技術含量的代碼,費時費力,而且還可能會賦值出錯,讓我們很頭疼。
MJExtension
架構就是為瞭解決這個問題而設計得第三方開源庫。這個開源庫是之前傳智部落格的講師李明傑老師寫的,現在他自己出來做了,我iOS入門都是看李明傑老師的培訓視頻學習的,他講得非常好,我非常喜歡他,他也算是我的老師了,他的作品我還是要學習下的。
提供了以下的一些方法實現:簡單的字典 –> 模型 JSON字串 –> 模型 複雜的字典 –> 模型 (模型裡麵包含了模型) 複雜的字典 –> 模型 (模型的數組屬性裡面又裝著模型) 複雜的字典 –> 模型(模型屬性名稱和字典的key不一樣) 字典數組 –> 模型數組 模型 –> 字典 模型數組 –> 字典數組 字典 –> CoreData模型 歸檔與解檔NSCoding 過濾字典的值
MJExtension
架構是利用Obj-C的運行時機制編寫的,現在iOS開發語言往Swift語言發展,我不太清楚Swift語言是否也有這種特性,該架構以後會不會在Swift語言上也發展下去不得而知,不過這個架構很輕量級,非常適合初級開發人員去看它的源碼,對理解Obj-C的運行時機制有非常大的協助。
二、Runtime運行時機制簡單瞭解
Runtime
簡稱運行時,就是系統在啟動並執行時候的一些機制,其中最主要的是訊息機制。
OC的函數調用類似於訊息發送,屬於動態調用過程。在編譯的時候並不能決定真正調用哪個函數。事實證明,在編譯階段,OC可以調用任何函數,即使這個函數並未實現,只要申明過就不會報錯。而C語言在編譯階段就會報錯。只有在真正啟動並執行時候才會根據函數的名稱找到對應的函數來調用。
例如,下面的這個代碼在編譯時間會被轉化:
/* OC方法調用 */[obj makeTest];/* 編譯時間Runtime會將上面的代碼轉為下面的訊息發送 */objc_msgSend(obj, @selector(makeText));
iOS的頂層基類NSObject含有一個指向objc_class結構體的isa指標:
@interface NSObject{ Class isa;};typedef struct objc_class *Class;struct objc_class { Class isa; // 指向metaclass,也就是靜態Class Class super_class ; // 指向其父類 const char *name ; // 類名 long version ; // 類的版本資訊,初始化預設為0 /* 一些標識資訊,如CLS_CLASS(0x1L)表示該類為普通class; CLS_META(0x2L)表示該類為metaclass */ long info; long instance_size ; // 該類的執行個體變數大小(包括從父類繼承下來的執行個體變數); struct objc_ivar_list *ivars; // 用於儲存每個成員變數的地址 /* 與info的一些標誌位有關,如是普通class則儲存物件方法,如是metaclass則儲存類方法; */ struct objc_method_list **methodLists ; struct objc_cache *cache; // 指向最近使用的方法的指標,用於提升效率; struct objc_protocol_list *protocols; // 儲存該類遵守的協議};
在
objc_msgSend
函數的調用過程:首先通過obj的isa指標找到obj對應的Class。 在Class中先去
cache
中通過SEL尋找對應函數
method
若
cache
中未找到,再去
methodLists
中尋找 若
methodLists
中未找到,則進入
superClass
按前面的步驟進行遞迴尋找 若找到
method
,則將
method
加入到
cache
中,以方便下次尋找,並通過
method
中的函數指標跳轉到對應的函數中去執行。 如果一直尋找到
NSObject
還沒尋找到,則會進入訊息動態處理流程。訊息動態處理流程:
/* 1. 時機處理之一,在這個方法中我們可以利用runtime的特性動態添加方法來處理 */+ (BOOL)resolveInstanceMethod:(SEL)sel;/* 2. 時機處理之二,在這個方法中看代理能不能處理,如果代理對象能處理,則轉接給代理對象 */- (id)forwardingTargetForSelector:(SEL)aSelector;/* 3. 訊息轉寄之一,該方法返回方法簽名,如果返回nil,則轉寄流程終止,拋出異常 */- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector;/* 4. 訊息轉寄之二,在該方法中我們可以對調用方法進行重新導向 */- (void)forwardInvocation:(NSInvocation *)anInvocation;
所以使用Runtime機制我們就可以動態向類添加方法或屬性:
/* 動態向一個類添加屬性 */class_addIvar(kclass, "expression", size, alignment, "*");/* 動態向一個類添加方法 */class_addMethod(kclass, @selector(setExpressionFormula:), (IMP)setExpressionFormula, "v@:@");class_addMethod(kclass, @selector(getExpressionFormula), (IMP)getExpressionFormula, "@@:");static void setExpressionFormula(id self, SEL cmd, id value){ NSLog(@"call setExpressionFormula");}static id getExpressionFormula(id self, SEL cmd){ NSLog(@"call getExpressionFormula"); return nil;}
v
表示void,
@
表示id類型,
:
表示SEL類型
"v@:@"
:表示傳回值為void,接受一個id類型、一個SEL類型、一個id類型的方法
"@@:"
:表示傳回值為id類型,接受一個id類型和一個SEL型別參數的方法
具體Runtime運行時使用細節,這裡就不細講,只是簡單瞭解下Runtime是可以做到動態向類添加屬性和方法就行。
三、MJExtension使用
MJExtension
的大部分方法實現都整合到了分類上,不需要使用新的類,只需要包含標頭檔MJExtension.h
即可。MJExtension
在github上的使用說明已經寫得十分明白了。
1. 簡單的字典 –> 模型模型類User定義:
typedef enum { SexMale, SexFemale} Sex;@interface User : NSObject@property (copy, nonatomic) NSString *name;/* 姓名 */@property (copy, nonatomic) NSString *icon;/* 頭像 */@property (assign, nonatomic) unsigned int age;/* 年齡 */@property (copy, nonatomic) NSString *height;/* 身高 */@property (strong, nonatomic) NSNumber *money;/* 資產 */@property (assign, nonatomic) Sex sex;/* 性別 */@property (assign, nonatomic, getter=isGay) BOOL gay;/* 是否是同性戀 */@end
使用執行個體:
NSDictionary *dict = @{ @"name" : @"Jack", @"icon" : @"lufy.png", @"age" : @20, @"height" : @"1.55", @"money" : @100.9, @"sex" : @(SexFemale),/* 枚舉需要使用NSNumber封裝 */ @"gay" : @"NO"};//字典轉模型,使用的是mj_objectWithKeyValues:方法User *user = [User mj_objectWithKeyValues:dict];
2. JSON字串 –> 模型使用執行個體:
// 定義一個JSON字串NSString*jsonString = @"{\"name\":\"Jack\", \"icon\":\"lufy.png\", \"age\":20}";// JSON字串轉模型User *user = [User mj_objectWithKeyValues:jsonString];
3. 複雜的字典 –> 模型 (模型裡麵包含了模型)模型類Status定義:
@interfaceStatus: NSObject@property(copy, nonatomic) NSString*text;@property(strong, nonatomic) User *user;/* 其他模型類型 */@property(strong, nonatomic) Status *retweetedStatus;/* 自我模型類型 */@end
使用執行個體:
NSDictionary *dict = @{ @"text" : @"Agree!Nice weather!", @"user" : @{ @"name" : @"Jack", @"icon" : @"lufy.png" }, @"retweetedStatus" : @{ @"text" : @"Nice weather!", @"user" : @{ @"name" : @"Rose", @"icon" : @"nami.png" } }};//字典轉模型,模型裡面含有模型Status *status = [Status mj_objectWithKeyValues:dict];NSString *text = status.text;NSString *name = status.user.name;NSString *icon = status.user.icon;NSLog(@"text=%@, name=%@, icon=%@", text, name, icon);// text=Agree!Nice weather!, name=Jack, icon=lufy.pngNSString *text2 = status.retweetedStatus.text;NSString *name2 = status.retweetedStatus.user.name;NSString *icon2 = status.retweetedStatus.user.icon;NSLog(@"text2=%@, name2=%@, icon2=%@", text2, name2, icon2);// text2=Nice weather!, name2=Rose, icon2=nami.png
4. 複雜的字典 –> 模型 (模型的數組屬性裡面又裝著模型)模型類Ad和StatusResult定義:
@interfaceAd: NSObject@property(copy, nonatomic) NSString*image;@property(copy, nonatomic) NSString*url;@end@interfaceStatusResult: NSObject/** 數組中儲存模型Status類型資料 */@property(strong, nonatomic) NSMutableArray*statuses;/** 數組中儲存模型Ad類型資料 */@property(strong, nonatomic) NSArray*ads;@property(strong, nonatomic) NSNumber*totalNumber;@end#import "MJExtension.h"/* 數組中儲存模型資料,需要說明數組中儲存的模型資料類型 */@implementation StatusResult/* 實現該方法,說明數組中儲存的模型資料類型 */+ (NSDictionary *)mj_ objectClassInArray{ return@{ @"statuses": @"Status",@"ads": @"Ad" };}@end
使用執行個體:
NSDictionary *dict = @{ @"statuses" : @[ @{ @"text" : @"Nice weather!", @"user" : @{ @"name" : @"Rose", @"icon" : @"nami.png" } }, @{ @"text" : @"Go camping tomorrow!", @"user" : @{ @"name" : @"Jack", @"icon" : @"lufy.png" } } ], @"ads" : @[ @{ @"image" : @"ad01.png", @"url" : @"http://www.ad01.com" }, @{ @"image" : @"ad02.png", @"url" : @"http://www.ad02.com" } ], @"totalNumber" : @"2014"};//字典轉模型,支援模型的數組屬性裡面又裝著模型StatusResult *result = [StatusResult mj_objectWithKeyValues:dict];//列印博主資訊for (Status *status in result.statuses) { NSString *text = status.text; NSString *name = status.user.name; NSString *icon = status.user.icon; NSLog(@"text=%@, name=%@, icon=%@", text, name, icon);}// text=Nice weather!, name=Rose, icon=nami.png// text=Go camping tomorrow!, name=Jack, icon=lufy.png//列印廣告for (Ad *ad in result.ads) { NSLog(@"image=%@, url=%@", ad.image, ad.url);}// image=ad01.png, url=http://www.ad01.com// image=ad02.png, url=http://www.ad02.com
5. 複雜的字典 –> 模型(模型屬性名稱和字典的key不一樣)模型類Bag和Student定義:
@interface Bag : NSObject@property (copy, nonatomic) NSString *name;@property (assign, nonatomic) double price;@end@interface Student : NSObject@property (copy, nonatomic) NSString *ID;@property (copy, nonatomic) NSString *desc;@property (copy, nonatomic) NSString *nowName;@property (copy, nonatomic) NSString *oldName;@property (copy, nonatomic) NSString *nameChangedTime;@property (strong, nonatomic) Bag *bag;@end#import "MJExtension.h"@implementation /* 設定模型屬性名稱和字典key之間的映射關係 */+ (NSDictionary *)mj_replacedKeyFromPropertyName{ /* 返回的字典,key為模型屬性名稱,value為轉化的字典的多級key */ return @{ @"ID" : @"id", @"desc" : @"desciption", @"oldName" : @"name.oldName", @"nowName" : @"name.newName", @"nameChangedTime" : @"name.info[1].nameChangedTime", @"bag" : @"other.bag" };}@end
使用執行個體:
NSDictionary *dict = @{ @"id" : @"20", @"desciption" : @"kids", @"name" : @{ @"newName" : @"lufy", @"oldName" : @"kitty", @"info" : @[ @"test-data", @{ @"nameChangedTime" : @"2013-08" } ] }, @"other" : @{ @"bag" : @{ @"name" : @"a red bag", @"price" : @100.7 } }};//字典轉模型,支援多級映射Student *stu = [Student mj_objectWithKeyValues:dict];//列印NSLog(@"ID=%@, desc=%@, oldName=%@, nowName=%@, nameChangedTime=%@", stu.ID, stu.desc, stu.oldName, stu.nowName, stu.nameChangedTime);// ID=20, desc=kids, oldName=kitty, nowName=lufy, nameChangedTime=2013-08NSLog(@"bagName=%@, bagPrice=%f", stu.bag.name, stu.bag.price);// bagName=a red bag, bagPrice=100.700000
6. 字典數組 –> 模型數組使用執行個體:
NSArray *dictArray = @[ @{ @"name" : @"Jack", @"icon" : @"lufy.png" }, @{ @"name" : @"Rose", @"icon" : @"nami.png" } ];//字典數組轉模型數組,使用的是mj_objectArrayWithKeyValuesArray:方法NSArray *userArray = [User mj_objectArrayWithKeyValuesArray:dictArray];//列印for (User *user in userArray) { NSLog(@"name=%@, icon=%@", user.name, user.icon);}// name=Jack, icon=lufy.png// name=Rose, icon=nami.png
7. 模型 –> 字典使用執行個體:
//建立一個模型對象User *user = [[User alloc] init];user.name = @"Jack";user.icon = @"lufy.png";Status *status = [[Status alloc] init];status.user = user;status.text = @"Nice mood!";//模型轉字典,使用的是mj_keyValues屬性NSDictionary*statusDict = status.mj_keyValues;NSLog(@"%@", statusDict);/*{text = "Nice mood!";user = {icon = "lufy.png";name = Jack;};}*/
8. 模型數組 –> 字典數組使用執行個體:
//建立模型數組User *user1 = [[User alloc] init];user1.name = @"Jack";user1.icon = @"lufy.png";User *user2 = [[User alloc] init];user2.name = @"Rose";user2.icon = @"nami.png";NSArray *userArray = @[user1, user2];//模型數組轉字典數組,使用的是mj_keyValuesArrayWithObjectArray:方法NSArray *dictArray = [User mj_keyValuesArrayWithObjectArray:userArray];NSLog(@"%@", dictArray);/* ( { icon = "lufy.png"; name = Jack; }, { icon = "nami.png"; name = Rose; } ) */
9. 字典 –> CoreData模型使用執行個體:
NSDictionary *dict = @{ @"name" : @"Jack", @"icon" : @"lufy.png", @"age" : @20, @"height" : @1.55, @"money" : @"100.9", @"sex" : @(SexFemale), @"gay" : @"true" };//字典轉為CoreData模型NSManagedObjectContext *context = nil;User *user = [User mj_objectWithKeyValues:dict context:context];[context save:nil];
10. 歸檔與解檔NSCoding模型類Bag添加實現:
@interface Bag : NSObject @property (copy, nonatomic) NSString *name;@property (assign, nonatomic) double price;@end#import "MJExtension.h"@implementation Bag//添加了下面的宏定義MJExtensionCodingImplementation/* 實現下面的方法,說明哪些屬性不需要歸檔和解檔 */+ (NSArray *)mj_ignoredCodingPropertyNames{ return @[@"name"];}@end
使用執行個體:
//建立模型Bag *bag = [[Bag alloc] init];bag.name = @"Red bag";bag.price = 200.8;//擷取歸檔路徑NSString *file = [NSHomeDirectory() stringByAppendingPathComponent:@"Desktop/bag.data"];//歸檔[NSKeyedArchiver archiveRootObject:bag toFile:file];//解檔Bag *decodedBag = [NSKeyedUnarchiver unarchiveObjectWithFile:file];NSLog(@"name=%@, price=%f", decodedBag.name, decodedBag.price);// name=(null), price=200.800000
11. 過濾字典的值模型類Book實現:
@interface Book: NSObject@property (copy, nonatomic) NSString *name;@property (strong, nonatomic) NSDate *publishedTime;@end#import "MJExtension.h"@implementation Book/* 轉化過程中對字典的值進行過濾和進一步轉化 */- (id)mj_newValueFromOldValue:(id)oldValue property:(MJProperty *)property{ if ([property.name isEqualToString:@"publisher"]) { if (oldValue == nil) { return @""; } } else if (property.type.typeClass == [NSDate class]) { NSDateFormatter *fmt = [[NSDateFormatter alloc] init]; fmt.dateFormat = @"yyyy-MM-dd"; return [fmt dateFromString:oldValue]; } return oldValue;}@end
使用執行個體:
NSDictionary *dict = @{ @"name" : @"5分鐘突破iOS開發", @"publishedTime" : @"2011-09-10" };//字典轉模型,過濾name為nil的情況,把NSString轉為NSDateBook *book = [Book mj_objectWithKeyValues:dict];//列印NSLog(@"name=%@, publishedTime=%@", book.name, book.publishedTime);
MJExtension
的github地址點這裡:CoderMJLee/MJExtension