MJExtension使用

转发自:MFExtension使用

一、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;};typedefstructobjc_class*Class;structobjc_class{Class isa;// 指向metaclass,也就是静态的ClassClass super_class ;// 指向其父类constchar*name ;// 类名longversion ;// 类的版本信息,初始化默认为0/* 一些标识信息,如CLS_CLASS(0x1L)表示该类为普通class;

      CLS_META(0x2L)表示该类为metaclass */longinfo;longinstance_size ;// 该类的实例变量大小(包括从父类继承下来的实例变量);structobjc_ivar_list*ivars;// 用于存储每个成员变量的地址/* 与info的一些标志位有关,如是普通class则存储对象方法,如是metaclass则存储类方法; */structobjc_method_list**methodLists;structobjc_cache*cache;// 指向最近使用的方法的指针,用于提升效率;structobjc_protocol_list*protocols;// 存储该类遵守的协议};

need-to-insert-img

在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;

need-to-insert-img

所以使用Runtime机制我们就可以动态向类添加方法或属性:

/* 动态向一个类添加属性 */class_addIvar(kclass,"expression", size, alignment,"*");/* 动态向一个类添加方法 */class_addMethod(kclass,@selector(setExpressionFormula:), (IMP)setExpressionFormula,"v@:@");class_addMethod(kclass,@selector(getExpressionFormula), (IMP)getExpressionFormula,"@@:");staticvoidsetExpressionFormula(idself, SEL cmd,idvalue){NSLog(@"call setExpressionFormula");  }staticidgetExpressionFormula(idself, SEL cmd)  {NSLog(@"call getExpressionFormula");returnnil;}

v表示void,@表示id类型,:表示SEL类型

"v@:@":表示返回值为void,接受一个id类型、一个SEL类型、一个id类型的方法

"@@:":表示返回值为id类型,接受一个id类型和一个SEL类型参数的方法

具体Runtime运行时使用细节,这里就不细讲,只是简单了解下Runtime是可以做到动态向类添加属性和方法就行。

三、MJExtension使用

MJExtension的大部分方法实现都集成到了分类上,不需要使用新的类,只需要包含头文件MJExtension.h即可。MJExtension在github上的使用说明已经写得十分明白了。

1. 简单的字典 --> 模型

模型类User定义:

typedefenum{    SexMale,    SexFemale} Sex;@interfaceUser:NSObject@property(copy,nonatomic)NSString*name;/* 姓名 */@property(copy,nonatomic)NSString*icon;/* 头像 */@property(assign,nonatomic)unsignedintage;/* 年龄 */@property(copy,nonatomic)NSString*height;/* 身高 */@property(strong,nonatomic)NSNumber*money;/* 资产 */@property(assign,nonatomic) Sex sex;/* 性别 */@property(assign,nonatomic,getter=isGay)BOOLgay;/* 是否是同性恋 */@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 = [Usermj_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"/* 数组中存储模型数据,需要说明数组中存储的模型数据类型 */@implementationStatusResult/* 实现该方法,说明数组中存储的模型数据类型 */+ (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 *statusinresult.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 *adinresult.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定义:

@interfaceBag:NSObject@property(copy,nonatomic)NSString*name;@property(assign,nonatomic)doubleprice;@end@interfaceStudent: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 *userinuserArray) {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添加实现:

@interfaceBag:NSObject@property(copy,nonatomic)NSString*name;@property(assign,nonatomic)doubleprice;@end#import"MJExtension.h"@implementationBag//添加了下面的宏定义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"];//归档[NSKeyedArchiverarchiveRootObject:bag toFile:file];//解档Bag *decodedBag = [NSKeyedUnarchiverunarchiveObjectWithFile:file];NSLog(@"name=%@, price=%f", decodedBag.name, decodedBag.price);// name=(null), price=200.800000

11. 过滤字典的值

模型类Book实现:

@interfaceBook:NSObject@property(copy,nonatomic)NSString*name;@property(strong,nonatomic)NSDate*publishedTime;@end#import"MJExtension.h"@implementationBook/* 转化过程中对字典的值进行过滤和进一步转化 */- (id)mj_newValueFromOldValue:(id)oldValue property:(MJProperty *)property{if([property.name isEqualToString:@"publisher"]) {if(oldValue ==nil) {return@"";        }            }elseif(property.type.typeClass == [NSDateclass]) {NSDateFormatter*fmt = [[NSDateFormatteralloc] init];        fmt.dateFormat =@"yyyy-MM-dd";return[fmt dateFromString:oldValue];    }returnoldValue;}@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

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,992评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,212评论 3 388
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,535评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,197评论 1 287
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,310评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,383评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,409评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,191评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,621评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,910评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,084评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,763评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,403评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,083评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,318评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,946评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,967评论 2 351

推荐阅读更多精彩内容