iOS RunTime解析

什么是Runtime

  • 我们的代码在运行过程中都会转化为runtime的C代码执行。如[target doSomething]都会被转化为 objc_msgSend(target, @selector(doSomething))
  • OC中一切都被设计成了对象,我们都知道一个类被初始化成一个实例,这个实例是一个对象。实际上一个类本质上也是一个对象,在runtime中用结构体表示。
  • 相关定义
// 描述类中的一个方法
typedef struct objc_method *Method;
// 实例变量
typedef struct objc_ivar *Ivar;
// 类别Category
typedef struct objc_category *Category;
// 类中声明的属性
typedef struct objc_property *objc_property_t;
  • 获取列表
    我们可以通过runtime的一系列方法获取类的一些信息(包括属性列表,方法列表,成员变量列表,和遵循的协议列表)。
unsigned int count;
    //获取属性列表
    objc_property_t *propertyList = class_copyPropertyList([self class], &count);
    for (unsigned int i=0; i<count; i++) {
        const char *propertyName = property_getName(propertyList[i]);
        NSLog(@"property---->%@", [NSString stringWithUTF8String:propertyName]);
    }

    //获取方法列表
    Method *methodList = class_copyMethodList([self class], &count);
    for (unsigned int i; i<count; i++) {
        Method method = methodList[i];
        NSLog(@"method---->%@", NSStringFromSelector(method_getName(method)));
    }

    //获取成员变量列表
    Ivar *ivarList = class_copyIvarList([self class], &count);
    for (unsigned int i; i<count; i++) {
        Ivar myIvar = ivarList[i];
        const char *ivarName = ivar_getName(myIvar);
        NSLog(@"Ivar---->%@", [NSString stringWithUTF8String:ivarName]);
    }

    //获取协议列表
    __unsafe_unretained Protocol **protocolList = class_copyProtocolList([self class], &count);
    for (unsigned int i; i<count; i++) {
        Protocol *myProtocal = protocolList[i];
        const char *protocolName = protocol_getName(myProtocal);
        NSLog(@"protocol---->%@", [NSString stringWithUTF8String:protocolName]);
    }

objc_msgSend消息发送

    //调用方法
     (
       (
        void (*)(id, SEL)//返回值void   (*)(任意类型,方法)
       )
       objc_msgSend
     )
     ( //对象,方法名
       [[Person alloc]init], sel_registerName("eat")
     );
    
    //调用传参
    (
        (
            void (*)(id, SEL, NSString *) //返回值void   (*)(任意类型,方法名,传入参数)
        )
        objc_msgSend
    )
    (   //对象,方法名,传入参数
        [[Person alloc] init], sel_registerName("eat:"), @"geyang"
    );
    
    //调用传参有返回值
    NSString *name =(
                        (
                            NSString* (*)(id, SEL, NSString *) //返回值NSString*   (*)(任意类型,方法名,传入参数)
                        )
                        objc_msgSend
                    )
                    (   //对象,方法名,传入参数
                        [[Person alloc] init], sel_registerName("whoeat:"), @"geyang"
                    );
    NSLog(@"%@", name);

Person.m

- (void)eat{
    NSLog(@"吃");
}

- (void)eat:(NSString *)name{
    NSLog(@"%@", name);
}

- (NSString *)whoeat:(NSString *)name{
    return name;
}

动态添加方法

  • 方法调用的过程
    1.首先,在相应操作的对象中的缓存方法列表中找调用的方法,如果找到,转向相应实现并执行。
    2.如果没找到,在相应操作的对象中的方法列表中找调用的方法,如果找到,转向相应实现执行。
    3.如果没找到,去父类指针所指向的对象中执行1,2.
    4.以此类推,如果一直到根类还没找到,转向拦截调用。
    5.如果没有重写拦截调用的方法,程序报错。
  • 拦截调用
    方法调用中说到没有找到方法会转向拦截调用。
    拦截调用就是,在找不到调用的方法程序奔溃之前,你有机会通过重写NSObject的四个方法来处理。
//调用一个不存在的 类方法触发
  +(BOOL)resolveClassMethod:(SEL)sel;
//调用一个不存在的 实例方法触发
 + (BOOL)resolveInstanceMethod:(SEL)sel;
/*
       重定向
       当resolveInstanceMethod方法没有处理,
       即返回NO或者直接返回了YES而没有添加方法,该方法被调用
   */
 - (id)forwardingTargetForSelector:(SEL)aSelector;
/*
       消息转发
       当重定向方法返回了nil,则methodSignatureForSelector会被调用
       系统会询问我们要一个合法的『类型编码(Type Encoding)』
   */
 - (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector;
//当实现了此方法后,-doesNotRecognizeSelector: 将不会被调用
// 在这里进行消息转发
 - (void)forwardInvocation:(NSInvocation *)anInvocation;
image.png
  • 实例
  MethodOne *method = [[MethodOne alloc]init];
//要求执行addMethod方法  但是类中没有声明该方法
 [method performSelector:@selector(addMethod) withObject:nil];

MethodOne.m

     /**
     动态添加eat方法
     第一个参数cls:给哪个类添加方法
     第二个参数name:添加方法的方法编号
     第三个参数imp:添加方法的函数实现(函数地址)
     第四个参数types:函数的类型
     v  :表示函数的返回值void
     @  :表示参数id self
     :  :表示SEL对象
     */
//没有找到addMethod方法后触发
 + (BOOL)resolveInstanceMethod:(SEL)sel{
    //判断名称是否一致(是否应没有addMethod方法而触发)
    if ([NSStringFromSelector(sel) isEqualToString:@"addMethod"]) {
        //动态添加一个方法dynamicAddMethod
        class_addMethod(self, sel, (IMP)dynamicAddMethod, "v@");
    }
    return YES;
}
//定义动态添加的方法
void dynamicAddMethod(id self, SEL _cmd, NSString *string){
    NSLog(@"这是动态添加的方法");
}
在没有找到`addMethod`方法时,
触发了`+ (BOOL)resolveInstanceMethod:(SEL)sel`方法 
我们动态的添加一个方法。
那么就会执行 dynamicAddMethod 方法

resolveInstanceMethod方法没有处理, 即返回NO或者直接返回了YES而没有添加方法,我们可以做重定向处理

//我们可以指定一个可以响应该方法的对象。返回self则会死循环。
 - (id)forwardingTargetForSelector:(SEL)aSelector{
    if (aSelector == @selector(addMethod:)) {
        //注意:Person类中定义了一个方法:-(void) addMethod;
        return [[Person alloc] init];
    }
    return [super forwardingTargetForSelector:aSelector];
}
那么就会执行 Person 类中的 addMethod 方法

重定向方法返回了nil,我们可以做最后一步处理:消息转发

//方法会被调用,系统会询问我们要一个合法的类型编码
//若返回 nil,则不会进入下一步,而是无法处理消息
 - (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector{
    return [NSMethodSignature signatureWithObjCTypes:"v@"];
}
//转发
 - (void)forwardInvocation:(NSInvocation *)anInvocation{
    //重新设置方法的选择器
    [anInvocation setSelector:@selector(jus)];
    //指定消息的接受者
    [anInvocation invokeWithTarget:self];
}
 - (void)jus{
    NSLog(@"这是转发后的执行输出");
}
那么就会执行jus方法

关联对象

现在你准备用一个系统的类,但是系统的类并不能满足你的需求,你需要额外添加一个属性。这种情况的一般解决办法就是继承。但是,只增加一个属性,就去继承一个类,总是觉得太麻烦类。这个时候,runtime的关联属性就发挥它的作用了。
首先创建一个UIButton的分类 UIButton+GYButton

UIButton+GYButton.h

#import <UIKit/UIKit.h>
//在分类中添加一个按钮点击事件的block回调
typedef void(^Click)(UIButton *sender);

@interface UIButton (GYButton)
@property (nonatomic, copy) Click click;
@property (nonatomic, strong) NSString *names;//添加属性
@end
UIButton+GYButton.m

//首先定义一个全局变量,用它的地址作为关联对象的key
static id obj;

//手动添加属性names的getter与setter方法
- (NSString *)names{
    return objc_getAssociatedObject(self, &name);
}
- (void)setNames:(NSString *)names{
    //设置关联对象
    objc_setAssociatedObject(self, &name, names, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
}

//按钮点击事件的block回调
- (void)setClick:(Click)click{
    /*
     object : 源对象
     key : 关联的键,objc_getAssociatedObject 方法通过不同的 key 即可取出对应的被关联对象
     value : 被关联的对象(当前要被替换的)
     policy : 是一个枚举值,表示关联对象的行为 copy/retain/nonatomic...
     */
    //设置关联对象
    objc_setAssociatedObject(self, &button, click, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
    if (click) {
        [self addTarget:self action:@selector(buttonClick:) forControlEvents:UIControlEventTouchUpInside];
    }
}
- (Click)click{
    return objc_getAssociatedObject(self, &button);
}
//按钮的点击事件
- (void)buttonClick:(UIButton *)sender{
    if (self.click) {
        self.click(sender);
    }
}

调用

//4.给分类添加属性
    UIButton *btn = [[UIButton alloc]initWithFrame:CGRectMake(30, 30, 50, 50)];
    btn.backgroundColor = [UIColor lightGrayColor];
    [btn setTitle:@"Click" forState:UIControlStateNormal];
    btn.names = @"123";
    NSLog(@"%@", btn.names);//输出123
    //按钮的点击事件 不需要再使用addTarget方法
    btn.click = ^(UIButton *sender){
        NSLog(@"点击了按钮:%@", sender.titleLabel.text);
    };

方法交换

方法交换,顾名思义,就是将两个方法的实现交换。例如,将A方法和B方法交换,调用A方法的时候,就会执行B方法中的代码,反之亦然。

首先创建一个UIImage的分类UIImage+GYImage

UIImage+GYImage.m

//load方法会在类第一次加载的时候被调用,调用的时间比较靠前,适合在这个方法里做方法交换
+ (void)load{
    //方法交换应该被保证,在程序中只会执行一次
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        //获取交换后方法
        Method GYimageName = class_getClassMethod(self, @selector(GYimageName:));
        
        //获取交换原的方法
        Method imageName = class_getClassMethod(self, @selector(imageNamed:));
        
        //交换方法 (新方法,旧方法)
        method_exchangeImplementations(GYimageName, imageName);
    });
}

// 不能在分类中重写系统方法imageNamed,因为会把系统的功能给覆盖掉,而且分类中不能调用super.
+ (instancetype)GYimageName:(NSString *)name{
    UIImage *image = [self GYimageName:name];
    if (image == nil) {
        NSLog(@"未加载到图片");
    }
    return image;
}

调用

UIImage *image = [UIImage imageNamed:@"123"];
//如果没有名为123的图片,则会打印出 “未加载到图片”。
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念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
  • 本文转载自:http://yulingtianxia.com/blog/2014/11/05/objective-...
    ant_flex阅读 744评论 0 1
  • 转载:http://yulingtianxia.com/blog/2014/11/05/objective-c-r...
    F麦子阅读 727评论 0 2
  • 我们常常会听说 Objective-C 是一门动态语言,那么这个「动态」表现在哪呢?我想最主要的表现就是 Obje...
    Ethan_Struggle阅读 2,170评论 0 7
  • 本文详细整理了 Cocoa 的 Runtime 系统的知识,它使得 Objective-C 如虎添翼,具备了灵活的...
    lylaut阅读 792评论 0 4