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

  1. JSON字符串 --> 模型
使用实例:
// 定义一个JSON字符串
NSString *jsonString = @"{\"name\":\"Jack\", \"icon\":\"lufy.png\", \"age\":20}";
// JSON字符串转模型
User *user = [User mj_objectWithKeyValues:jsonString];
  1. 复杂的字典 --> 模型 (模型里面包含了模型)
模型类Status定义:
@interface Status : 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.png
NSString *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

  1. 复杂的字典 --> 模型 (模型的数组属性里面又装着模型)
    模型类Ad和StatusResult定义:
@interface Ad : NSObject
@property (copy, nonatomic) NSString *image;
@property (copy, nonatomic) NSString *url;
@end

@interface StatusResult : 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

  1. 复杂的字典 --> 模型(模型属性名和字典的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-08
NSLog(@"bagName=%@, bagPrice=%f", stu.bag.name, stu.bag.price);
// bagName=a red bag, bagPrice=100.700000

  1. 字典数组 --> 模型数组
使用实例:
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

  1. 模型 --> 字典
使用实例:
//创建一个模型对象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; }; } */

  1. 模型数组 --> 字典数组
使用实例:
//创建模型数组
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; } ) */

  1. 字典 --> 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];
  1. 归档与解档NSCoding
模型类Bag添加实现:
@interface Bag : NSObject <NSCoding>
@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

  1. 过滤字典的值
模型类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转为NSDate
Book *book = [Book mj_objectWithKeyValues:dict];
//打印
NSLog(@"name=%@, publishedTime=%@", book.name, book.publishedTime);

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

推荐阅读更多精彩内容

  • 转至元数据结尾创建: 董潇伟,最新修改于: 十二月 23, 2016 转至元数据起始第一章:isa和Class一....
    40c0490e5268阅读 1,678评论 0 9
  • 对于从事 iOS 开发人员来说,所有的人都会答出【runtime 是运行时】什么情况下用runtime?大部分人能...
    梦夜繁星阅读 3,697评论 7 64
  • 1.简单的字典 --> 模型 ----核心代码 mj_objectWithKeyValues:2.JSON字符...
    MaybeLove00阅读 972评论 0 0
  • MJExtension的github地址 1. Getting Started【开始使用】 只需要一行代码,就能实...
    iYeso阅读 982评论 0 2
  • 曾有几位实业界的朋友跟我说,“我在推一个品牌,钱砸了不少,但好像没什么效果,继续吧,担心是个无底洞,放弃吧,又不甘...
    颜伯卿阅读 538评论 0 1