Objective-C Runtime介绍与应用示例

在看一些牛逼闪闪的开源框架时发现都使用了Runtime黑魔法,那么究竟什么是Runtime呢?我们都知道Objective-C是一门动态语言,Objective-C最大的特色是承自Smalltalk的消息传递模型(message passing),这种机制和当今C++式的主流风格差异甚大,C++里类别与方法的关系严格清楚,而在Objective-C中,类别与消息的关系比较松散,调用方法视为对对象发送消息,所有方法都被视为对消息的回应。所有消息处理直到运行时(runtime)才会动态决定,并交由类别自行决定如何处理收到的消息。简单地说,Runtime系统是一个包含由一系列函数和数据结构组成公共接口的动态共享库,在头文件位于/usr/include/objc.的目录。当你编写objective - C代码的时候,这些函数允许您使用纯C代码去复制,并在编译时执行这些函数。

Objective-C程序与runtime system的交互体现在三个不同的层次:
1.通过Objective-C源代码;
2.通过Foundation frameworkNSObject类定义的方法;
3.通过直接调用runtime函数。

一:Runtime最主要的就是消息机制。我们从The objc_msgSend Function——消息发送开始说起。

[receiver message]

objective - c中,直到运行时消息才会和对应实现的方法绑定。消息调用时的转换是在编译期间进行的。上面的代码实际上会被编译器转化为:

objc_msgSend(receiver, selector)

objc_msgSend是一个消息发送传递的函数,。这个函数需要消息接收者和消息方法名(这个方法名选择器) 作为它的两个主要参数。

objc_msgSend(receiver, selector, arg1, arg2, ...)

消息传入的任何参数也会交给objc_msgSend。

objc/message.h文件中可以看到objc_msgSend 函数的定义:

OBJC_EXPORT id objc_msgSend(id self, SEL op, ...)

SEL
objc_msgSend函数第二个参数类型为SEL,它是selectorOC中的表示类型。selector是方法选择器,可以理解为区分方法的ID,而这个 ID的数据结构是SEL:它被定义在objc/objc.h目录下:

/// An opaque type that represents a method selector.
typedef struct objc_selector *SEL;

id
objc_msgSend函数第一个参数类型为id,与SEL 一样,id也被定义在 objc/objc.h 目录下:

/// A pointer to an instance of a class.
typedef struct objc_object *id;

id 是一个结构体指针类型,它可以指向 Objective-C中的任何对象。objc_object结构体定义如下:

/// Represents an instance of a class.
struct objc_object {
    Class isa  OBJC_ISA_AVAILABILITY;
};

objc_object结构体包含一个isa指针,根据isa指针就可以顺藤摸瓜找到对象所属的类。

Class
之所以说isa是指针是因为Class其实是一个指向objc_class结构体的指针:

/// An opaque type that represents an Objective-C class.
typedef struct objc_class *Class;

那么objc_class的庐山真面目又是什么呢?进入objc/runtime.h我们便可一窥究竟:

struct objc_class {
    Class isa  OBJC_ISA_AVAILABILITY;

#if !__OBJC2__
    Class super_class                                        OBJC2_UNAVAILABLE;
    const char *name                                         OBJC2_UNAVAILABLE;
    long version                                             OBJC2_UNAVAILABLE;
    long info                                                OBJC2_UNAVAILABLE;
    long instance_size                                       OBJC2_UNAVAILABLE;
    struct objc_ivar_list *ivars                             OBJC2_UNAVAILABLE;
    struct objc_method_list **methodLists                    OBJC2_UNAVAILABLE;
    struct objc_cache *cache                                 OBJC2_UNAVAILABLE;
    struct objc_protocol_list *protocols                     OBJC2_UNAVAILABLE;
#endif

} OBJC2_UNAVAILABLE;

这就是我们常说的类:
·Class 也有一个 isa 指针,指向其所属的元类(meta).
·super_class:指向其超类.
·name:是类名.
·version:是类的版本信息.
·info:类的详情.
·instance_size:是该类的实例对象的大小.
·ivars:指向该类的成员变量列表.
·methodLists:指向该类的实例方法列表,它将方法选择器和方法实现地址联系起来。methodLists 是指向 ·objc_method_list 指针的指针,也就是说可以动态修改 *methodLists 的值来添加成员方法,这也是Category实现的原理,同样解释了Category不能添加属性的原因.
·cache:Runtime系统会把被调用的方法存到 cache中(理论上讲一个方法如果被调用,那么它有可能今后还会被调用),下次查找的时候效率更高.
·protocols:指向该类的协议列表.

总结一下:
首先,Runtime 系统会把方法调用转化为消息发送,即 objc_msgSend,并且把方法的调用者,和方法选择器,当做参数传递过去.此时,方法的调用者会通过isa 指针来找到其所属的类,然后在cache 或者methodLists 中查找该方法,找得到就跳到对应的方法去执行.
如果在类中没有找到该方法,则通过super_class往上一级超类查找(如果一直找到NSObject都没有找到该方法的话,就会调用下面这几个方法,给你“补救”的机会,你可以先理解为几套防止程序crash的备选方案,我们就是利用这几个方案进行消息转发,注意一点,前一套方案实现后一套方法就不会执行。如果这几套方案你都没有做处理,那么程序就会报错crash

方案一:
+ (BOOL)resolveInstanceMethod:(SEL)sel
+ (BOOL)resolveClassMethod:(SEL)sel
方案二:
- (id)forwardingTargetForSelector:(SEL)aSelector

方案三:

- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector;
- (void)forwardInvocation:(NSInvocation *)anInvocation;

示意图:

前面我们说 methodLists指向该类的实例方法列表,实例方法即-方法,那么类方法(+方法)存储在哪儿呢?类方法被存储在元类中,Class通过 isa 指针即可找到其所属的元类。

上图实线是 super_class 指针,虚线是 isa 指针。根元类的超类是NSObject,而 isa 指向了自己。NSObject 的超类为 nil,也就是它没有超类。

介绍完理论,下面让我们开始真正在开发中使用objc_msgSend吧!
1.创建Fish类并添加方法:

@interface Fish : NSObject
//游泳
+ (void)swim;
- (void)swim;

//吐泡泡
- (void)bloweBubbles:(int)num;
@end

2.使用objc_msgSend

//类对象发送消息
- (void)testClassObject{
    //获取类对象
    Class fClass = [Fish class];
    //运行时
    objc_msgSend(fClass, @selector(bloweBubbles:),100);
}

//对象发送消息
- (void)testObject{
    
    Fish * fish = [[Fish alloc]init];
    //    运行时,发送消息
    objc_msgSend(fish, @selector(swim));
    
    //    带参数
    objc_msgSend(fish, @selector(bloweBubbles:),10);
}

二:Method Swizzling
Method Swizzing是发生在运行时的,主要用于在运行时将两个方法进行交换。当我们在开发中发现系统自带的方法功能不够,需要给系统自带的方法扩展一些功能,并且保持原有的功能时,Method Swizzing便派上用场了。
用法示例:比如我们想给UIImageimageNamed:方法增加图片加载是否成功的提示。

1.创建UIImage的分类并声明作为交换的方法:

#import <UIKit/UIKit.h>

@interface UIImage (JF)

+ (__kindof UIImage *)imageWithName:(NSString *)name;

@end

2.实现Method Swizzing

#import "UIImage+JF.h"
#import <objc/message.h>
@implementation UIImage (JF)
+ (void)load
{
   // 交换方法
    // 获取imageWithName方法
    Method imageWithName = class_getClassMethod(self, @selector(imageWithName:));
    
    // 获取imageNamed方法
    Method imageNamed = class_getClassMethod(self, @selector(imageNamed:));
    
    // 交换方法,相当于交换函数地址
    method_exchangeImplementations(imageWithName, imageNamed);
    
}

// 既能加载图片又能提示是否加载成功
+ (__kindof UIImage *)imageWithName:(NSString *)name
{
  // 这里调用imageWithName,相当于调用imageName
    UIImage * image = [self imageWithName:name];
  //用打印模拟提示功能
    if (image == nil) {
        NSLog(@"图片加载失败");
    }
    return image;
}

三:动态方法处理
有时候,我们可能需要提供一个动态的实现方法,具体如下:
创建Fish类,动态添加drink:方法

#import "Fish.h"
@implementation Fish

//定义函数
//没有返回值,参数(id,SEL,id)
//void(id,SEL,id)
void drink(id self,SEL _cmd,id param1)
{
    NSLog(@"鱼儿%@,%@,%@口水",self,NSStringFromSelector(_cmd),param1);
}


//动态添加方法首先实现resolveInstanceMethod
/*
 resolveInstanceMethod调用:当调用了没有实现的方法,就会调用该方法resolveInstanceMethod
 resolveInstanceMethod的作用:知道哪些方法没有实现,从而动态添加方法
 sel:没有实现的方法的编码
 */
+ (BOOL)resolveInstanceMethod:(SEL)sel{
    
    //动态添加drink方法
    if( sel == @selector(drink:))
    {
        /*
         cls:给哪个类添加方法
         SEl:添加方法的方法编号
         IMP:方法实现,函数入口,函数名
         types:方法类型
         */
        class_addMethod(self, sel, (IMP)drink, "v@:@");
        
    }
        
        return YES;
}
- (void)viewDidLoad {
    [super viewDidLoad];

    Fish * f = [[Fish alloc]init];
    [f performSelector:@selector(drink:) withObject:@10];
}

** 四:动态添加属性 **
在一般情况下,我们知道在分类中是无法添加属性的,因为@property在分类中,只会生成get,set方法的声明,不会生成下划线成员属性,和get,set方法的实现。但是,通过Runtime我们可以实现动态地添加属性。

1.创建NSObject分类

#import <Foundation/Foundation.h>
@interface NSObject (JF)

@property(nonatomic,strong)NSString * name;

@end

#import "NSObject+JF.h"
#import <objc/message.h>

// 定义关联的key
static const char *key = "name";

@implementation NSObject (JF)

- (void)setName:(NSString *)name{
    
    // 第一个参数:给哪个对象添加关联
    // 第二个参数:关联的key,通过这个key获取
    // 第三个参数:关联的value
    // 第四个参数:关联的策略
    objc_setAssociatedObject(self, key, name, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
    
    
}

- (NSString *)name{
    
    // 根据关联的key,获取关联的值。
    return objc_getAssociatedObject(self, key);
}

** 五:Runtime字典转模型 **
1.遍历模型中所有属性
2.给模型中的每个属性赋值

#import "NSObject+JFExtension.h"
#import <objc/message.h>

@implementation NSObject (JFExtension)

+ (instancetype)modelWithDic:(NSDictionary *)dic{
    
    // 思路:遍历模型中所有属性-》使用运行时
    // 0.创建对应的对象
    id objc = [[self alloc]init];
    
    // 1.利用runtime给对象中的成员属性赋值
    unsigned int count;
    // 获取类中的所有成员属性
    Ivar * ivarList = class_copyIvarList(self, &count);
    
    for (int i = 0; i < count; i++) {
        // 根据角标,从数组取出对应的成员属性
        Ivar ivar = ivarList[i];
        
        // 获取成员属性名
        NSString * propertyName = [NSString stringWithUTF8String:ivar_getName(ivar)];
        
        // 处理成员属性名->字典中的key
        // 从第一个角标开始截取
        NSString * key = [propertyName substringFromIndex:1];
        
        // 根据成员属性名去字典中查找对应的value
        id value = dic[key];
        
        // 获取成员属性类型
        NSString * propertyType = [NSString stringWithUTF8String:ivar_getTypeEncoding(ivar)];
        
        // 二级转换:如果字典中还有字典,也需要把对应的字典转换成模型
        // 判断下value是否是字典
        if ([value isKindOfClass:[NSDictionary class]]) {
            // 字典转模型
            // 1.获取模型的类对象,调用modelWithDict
            // 2.模型的类名已知,就是成员属性的类型
            
            // 获取成员属性类型
            // 生成的是这种@"@\"User\"" 类型 -》 @"User"  在OC字符串中 \" -> ",\是转义的意思,不占用字符
            // 裁剪类型字符串
            NSRange range = [propertyType rangeOfString:@"\""];
            
            propertyType = [propertyType substringFromIndex:range.location + range.length];
            
            range = [propertyType rangeOfString:@"\""];
            
            // 裁剪到哪个角标,不包括当前角标
            propertyType = [propertyType substringToIndex:range.location];
            
            // 根据字符串类名生成类对象
            Class modelClass = NSClassFromString(propertyType);
            
            // 有对应的模型才需要转
            if (modelClass) {
               // 把字典转模型
                value = [modelClass modelWithDic:value];
            }
            
            // 三级转换:NSArray中也是字典,把数组中的字典转换成模型.
            // 判断值是否是数组
        }else if ([value isKindOfClass:[NSArray class]]){
            
            // 判断对应类有没有实现字典数组转模型数组的协议
            if ([self respondsToSelector:@selector(arrayContainModelClass)]) {
                
                // 转换成id类型,就能调用任何对象的方法
                id idSelf = self;
                
                // 获取数组中字典对应的模型
                NSString *type = [idSelf arrayContainModelClass][key];
                
                // 生成模型
                Class classModel = NSClassFromString(type);
                
                NSMutableArray *muArray = [NSMutableArray array];
                
                // 遍历字典数组,生成模型数组
                for (NSDictionary * dic in value) {
                    
                    // 字典转模型
                    id model = [classModel modelWithDic:dic];
                    [muArray addObject:model];
                }
                
                // 把模型数组赋值给value
                value = muArray;
                
            }
        }
        
        // 有值,才需要给模型的属性赋值
        // 利用KVC给模型中的属性赋值
        if (value) {
            [objc setValue:value forKey:key];
        }
        
    }
    
    return objc;
}

@end

完整字典转模型代码请点这里这里

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

推荐阅读更多精彩内容

  • 转至元数据结尾创建: 董潇伟,最新修改于: 十二月 23, 2016 转至元数据起始第一章:isa和Class一....
    40c0490e5268阅读 1,678评论 0 9
  • 本文详细整理了 Cocoa 的 Runtime 系统的知识,它使得 Objective-C 如虎添翼,具备了灵活的...
    lylaut阅读 791评论 0 4
  • 文中的实验代码我放在了这个项目中。 以下内容是我通过整理[这篇博客] (http://yulingtianxia....
    茗涙阅读 907评论 0 6
  • 转载:http://yulingtianxia.com/blog/2014/11/05/objective-c-r...
    F麦子阅读 724评论 0 2
  • 游居在高原的丘陵沟壑之城已经一年四个月十八天,仔细算来还需再加十五天宝塔山下的生活,延安这个地方是英雄的城孕...
    流曦溪阅读 301评论 0 0