iOS开发runtime介绍及常用功能实例

1、消息发送机制
objc语法中类似[object hello]的方法调用语法,在运行时会转换成c语言的objc_msgSend函数,并默认传入两个参数:(id self,SEl sel),参数id 传递方法调用对象,sel传递方法选择器。因此,我们可以通过运行时直接调用objc_msgSend函数调用方法(包括私有方法)。
objc_msgSend函数做了动态绑定所需要的一切:①它首先找到方法选标所对应的方法实现。因为不同的类对同一方法可能会有不同的实现,所以找到的方法实现依赖于消息接收者的类型。②然后将消息接收者的对象以及方法中指定的参数传给找到的方法实现。③将方法实现的返回值作为该函数的返回值返回。

当一个方法会被连续调用很多次,并且你希望节省每次调用都要发送消息的开销时,可以使用方法地址来调用就显得很有效果。
ps:首先要到build Settings关闭 strict checking,否则会报错。

- (void)sendMsg{
    Person *p = [Person new];
    //调用类方法
    objc_msgSend([Person class],@selector(run));
    //调用对象方法
    objc_msgSend(p, @selector(run));
}

//Person.m文件中定义两个私有方法
+ (void)run{
    NSLog(@"调用了run类方法");
}
- (void)run{
    NSLog(@"调用了run实例方法");
}

2、交换方法
当对象调用方法时,会先到所对应的类中匹配方法列表,若没有则往父类找,一直没找到的话就报unrecognize selector的错误。但父类的方法列表中并不记录方法的具体内容,它记录的是方法实现(IMP)的入口。我们可以通过runtime交换方法的实现,从而达到调a方法名,但对应的是b的实现的效果。

假如有如下场景:iOS7刚流行时,项目过程中,产品经理本来说全部采用扁平化的图片,但后来又要求为iOS6之前的配备拟物化的图片。此时,你的图片设置逻辑已经写了很多了,逐个去查找修改肯定不现实。
解决方案:可以为image添加一个分类,分类中自定义一个方法,内部做系统版本的逻辑判断,最后仍调用系统的方法产生图片。然后使用运行时交换自定义方法和系统原本方法来达到不改动源代码达到目的的效果。

//分类中
#import "UIImage+Exchange.h"
#import <objc/runtime.h>
@implementation UIImage (Exchange)
/**
 *  为了让方法交换一开始就生效,把方法交换的代码写在load方法里。load方法在程序一启动时就生效。
 */
+ (void)load{
    //获取原生的方法
    Method imageNamed = class_getClassMethod([UIImage class], @selector(imageNamed:));
    //获取自定义的方法
    Method imageWithName = class_getClassMethod([UIImage class], @selector(imageWithName:));
    //交换方法
    method_exchangeImplementations(imageNamed, imageWithName);
}
/**
 *  自定义图片类方法
 *
 *  @param name 图片名字
 *
 *  @return 返回图片实例
 */
+ (UIImage *)imageWithName:(NSString *)name{
    //判断系统版本
    CGFloat version =  [[UIDevice currentDevice].systemVersion floatValue];
    if (version > 6.0) {
        NSLog(@"返回扁平化的图片");
        //此处做相应逻辑处理,如修改资源名字
    }else{
        NSLog(@"返回拟物化的图片");
    }
    //调用系统的方法生成对应的图片。因为已经交换了方法,所以使用imageWithName才能掉到系统方法
    [self imageWithName:name];
//返回符合要求的图片
    return nil;
}

3、返回实例变量
什么是实例变量? 使用属性时自动帮你生成的带下划线的,如_age
应用场景:如字典转模型

- (void)getVarList{
    //类中实例变量的数量。函数返回一个数组,还有一个返回值通过传地址进去返回。
    unsigned int ivarCount;
    Ivar* ivarList =  class_copyIvarList([Person class], &ivarCount);
    //遍历返回的实例变量数组
    for (int i = 0; i < ivarCount; i++) {
        //打印实例变量名
        NSLog(@"%s",ivar_getName(ivarList[i]));
    }
}

4、获取方法列表,类似获取实例变量

- (void)getMethodList{
    unsigned int outCount;
    Method *methodList = class_copyMethodList([Person class], &outCount);
    for (int i = 0; i < outCount;  i++) {
        SEL sel = method_getName(methodList[i]);
        NSLog(@"%@",NSStringFromSelector(sel));
    }
}

5、通过runtime动态创建类并添加实例变量和方法(kvo底层用到了这个,动态创建了派生类,并重写了set方法)

- (void)addClass{
    //分别为 继承父类的类名 ,类名 ,
    Class myClass = objc_allocateClassPair([NSObject class], "City", 0);
    //添加实例变量,分别为变量所需类名,变量名,变量类型大小,变量类型编码
    BOOL flag =  class_addIvar([myClass class], "_name", sizeof(NSString *), 0, "@");
    if (flag) {
        //如果添加成功,使用kvc为变量赋值
        id newCity = [[myClass alloc]init];
        [newCity setValue:@"深圳" forKey:@"_name"];
        //读取变量值
        NSLog(@"%@",[newCity valueForKey:@"_name"]);
    }
    //添加方法,方法所属类,方法选择器,方法实现,方法实现的类型编码
    //类型编码要参考官方文档,v标识void返回类型 ,
    //调用方法时,默认传入 调用者对象和方法选择器:对应 @:
    //第三个参数为NSString对象,所以用 @,综合为 v@:@
    flag = class_addMethod([myClass class], @selector(aMethodSel:),(IMP)aMethodImp, "v@:@");
    //调用方法
    id newCity = [[myClass alloc]init];
    [newCity aMethodSel:@"添加方法"];
}

//方法实现,方法选择器最终找到方法实现并执行里面的代码
void aMethodImp(id self,SEL sel, NSString *name){
      NSLog(@"%s  methodname=%@",__func__,name);
}

//方法选择器,类似于声明,里面的内容并不会执行
- (void)aMethodSel:(NSString *)name{
     NSLog(@"%s  selname=%@",__func__,name);
}

6、为分类添加属性,有时候为了调用方便,采用分类的形式,但又需要保存某些值,可以使用这个方法

//分类中定义一个url属性,重写set,get
- (void)setUrl:(NSString *)url{
    /**
     *  关联属性的对象
        属性的key值,这个值名字可以随便设,和取值时候保持一致就行
        属性的value值
        //内存管理策略,如copy,assign等
     */
    objc_setAssociatedObject(self, "url", url, OBJC_ASSOCIATION_COPY_NONATOMIC);
}

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

推荐阅读更多精彩内容

  • 转至元数据结尾创建: 董潇伟,最新修改于: 十二月 23, 2016 转至元数据起始第一章:isa和Class一....
    40c0490e5268阅读 1,715评论 0 9
  • objc_getAssociatedObject返回与给定键的特定对象关联的值。ID objc_getAssoci...
    有一种再见叫青春阅读 1,582评论 0 7
  • 这篇文章完全是基于南峰子老师博客的转载 这篇文章完全是基于南峰子老师博客的转载 这篇文章完全是基于南峰子老师博客的...
    西木阅读 30,557评论 33 466
  • 参数自一个指针,指向类的要接收消息的实例。 OP在处理该信息的方法的选择。 ......可变参数列表包含参数的方法...
    reallychao阅读 805评论 0 0
  • 家里无论发生了什么大事,总是瞒着最小的我,他们用“小孩别管大人事”“告诉你你也帮不上忙”这种借口来搪塞我。我,已经...
    永远八岁阅读 199评论 0 0