浅谈 Method Swizzling 中遇到的一些问题

如果对 Runtime 有一定了解的话,一定听说过或者用过这个函数:

OBJC_EXPORT void method_exchangeImplementations(Method m1, Method m2) 

它通常叫做 Method Swizzling,算是objc的 “黑魔法” 了,作用就是在程序运行期间动态的给两个方法互换实现。

最近有用到这个,总结下遇到的一些问题:

静态(类)方法和实例方法的交换实现方式一样吗?

交换静态(类)方法的正确姿势:

+ (void)load {
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        SEL s1 = @selector(go);
        SEL s2 = @selector(stop);
        Class class = object_getClass((id)self);
        Method m1 = class_getClassMethod(class, s1);
        Method m2 = class_getClassMethod(class, s2);
        BOOL success = class_addMethod(class, s1, method_getImplementation(m2), method_getTypeEncoding(m2));
        if (success){
            class_replaceMethod(class, s2, method_getImplementation(m1), method_getTypeEncoding(m1));
        }
        else{
            method_exchangeImplementations(m1, m2);
        }
    });
}

+ (void)go {
    NSLog(@"Go!");
}

+ (void)stop {
    NSLog(@"Stop!");
}

交换实例方法的正确姿势:

+ (void)load {
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        SEL s1 = @selector(go);
        SEL s2 = @selector(stop);
        Class class = [self class];
        Method m1 = class_getInstanceMethod(class, s1);
        Method m2 = class_getInstanceMethod(class, s2);
        BOOL success = class_addMethod(class, s1, method_getImplementation(m2), method_getTypeEncoding(m2));
        if (success){
            class_replaceMethod(class, s2, method_getImplementation(m1), method_getTypeEncoding(m1));
        }
        else{
            method_exchangeImplementations(m1, m2);
        }
    });
}

- (void)go {
    NSLog(@"Go!");
}

- (void)stop {
    NSLog(@"Stop!");
}

我们可以发现上面两段方法的区别在于:

静态(类)方法的ClassMethod是这样的:

Class class = object_getClass((id)self);
Method m1 = class_getClassMethod(class, s1);
Method m2 = class_getClassMethod(class, s2);

实例方法的ClassMethod是这样的:

Class class = [self class];
Method m1 = class_getInstanceMethod(class, s1);
Method m2 = class_getInstanceMethod(class, s2);

Runtime 中class_getClassMethod的实现:

Method class_getClassMethod(Class cls, SEL sel)
{
    if (!cls  ||  !sel) return nil;

    return class_getInstanceMethod(cls->getMeta(), sel);
}

大家可以下载 Runtime 源码 看看。

其实class_getClassMethod内就是调了下class_getInstanceMethod,只是传的Class参数不一样:

  • class_getClassMethodClass参数传的是元类,也就是类对象的类。(关于元类大家可以看看这篇翻译的文章 Objective-C 中的元类(meta class)是什么?
  • class_getInstanceMethodClass 参数看名字就可以理解,既然是获得实例方法,自然传的就是实例对象的类。

object_getClass(id obj) 又是什么呢?还是看源码:

Class object_getClass(id obj)
{
    if (obj) return obj->getIsa();
    else return Nil;
}

object_getClass() 就是顺着 isa 的指向链找到对应的类。

有兴趣的可以看看这篇文章:为什么 object_getClass(obj) 与 [OBJ class] 返回的指针不同

静态(类)方法和实例方法里面的 self 表示的含义一样吗?

先说明下为什么我会突然有这个疑问:

当初为了方便测试交换两个静态方法的实现,我直接撸了这一段代码:

- (void)viewDidLoad {
    [super viewDidLoad];
    
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        SEL s1 = @selector(test1);
        SEL s2 = @selector(test2);
        Class class = object_getClass((id)self);
        Method m1 = class_getClassMethod(class, s1);
        Method m2 = class_getClassMethod(class, s2);
        BOOL success = class_addMethod(class, s1, method_getImplementation(m2), method_getTypeEncoding(m2));
        if (success){
            class_replaceMethod(class, s2, method_getImplementation(m1), method_getTypeEncoding(m1));
        }
        else{
            method_exchangeImplementations(m1, m2);
        }
    });

    [ViewController test1];
    [ViewController test2];
}

+ (void)test1 {
    NSLog(@"test1");
}

+ (void)test2 {
    NSLog(@"test2");
}

打印结果如下:

2017-01-25 17:13:53.356 RuntimeDemo[16180:4689887] test1
2017-01-25 17:13:53.356 RuntimeDemo[16180:4689887] test2

居然交换失败了!

然后我尝试着把代码挪到 + (void)load{} 里面:

+ (void)load {
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        SEL s1 = @selector(test1);
        SEL s2 = @selector(test2);
        Class class = object_getClass((id)self);
        Method m1 = class_getClassMethod(class, s1);
        Method m2 = class_getClassMethod(class, s2);
        BOOL success = class_addMethod(class, s1, method_getImplementation(m2), method_getTypeEncoding(m2));
        if (success){
            class_replaceMethod(class, s2, method_getImplementation(m1), method_getTypeEncoding(m1));
        }
        else{
            method_exchangeImplementations(m1, m2);
        }
    });
}

- (void)viewDidLoad {
    [super viewDidLoad];

    [ViewController test1];
    [ViewController test2];
}

打印结果如下:

2017-01-25 17:23:32.739 RuntimeDemo[16212:4705811] test2
2017-01-25 17:23:32.739 RuntimeDemo[16212:4705811] test1

可以发现交换成功了!

在找原因之前我们先来看一张经典的图:

从左到右依次是:实例对象、类对象、元类(类对象的类)。

我们通常这样来获取这三个对象:

Person *obj = [Person new];
NSLog(@"instance         :%p", obj);
NSLog(@"class            :%p", object_getClass(obj));
NSLog(@"meta class       :%p", object_getClass(object_getClass(obj)));

我们来大胆猜想下:在实例方法中,self表示的是实例对象这个大家都知道,那在类(静态)方法中self表示的是不是就是实例对象的类,也就是类对象呢?我们直接撸代码来验证下:

- (void)viewDidLoad {
    [super viewDidLoad];
    
    [self test1];
    [ViewController test2];
}

- (void)test1 {
    NSLog(@"%p",self);
    NSLog(@"%p",object_getClass((id)self));
}

+ (void)test2 {
    NSLog(@"%p",self);
}

打印结果如下:

2017-01-25 22:54:37.325 RuntimeDemo[3841:159827] 0x7ff6ee50ac00
2017-01-25 22:54:38.752 RuntimeDemo[3841:159827] 0x100ec0fe0
2017-01-25 22:54:42.740 RuntimeDemo[3841:159827] 0x100ec0fe0

果然不出所料,我们的大胆猜想是正确的。

回过头,我们修改下实例方法中交换的实现:

- (void)viewDidLoad {
    [super viewDidLoad];
    
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        ····
        
        Class class = object_getClass(object_getClass((id)self));
      
        ····
       }
    });

    [ViewController test1];
    [ViewController test2];
}

打印结果也自然而然的交换成功了:

2017-01-25 23:02:12.579 RuntimeDemo[3926:164600] test2
2017-01-25 23:02:12.580 RuntimeDemo[3926:164600] test1

网上的 Method Swizzling 有两种写法,到底哪种靠谱?

两种写法分别如下:

+ (void)load {
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        SEL s1 = @selector(go);
        SEL s2 = @selector(stop);
        Class class = object_getClass((id)self);
        Method m1 = class_getClassMethod(class, s1);
        Method m2 = class_getClassMethod(class, s2);
        BOOL success = class_addMethod(class, s1, method_getImplementation(m2), method_getTypeEncoding(m2));
        if (success){
            class_replaceMethod(class, s2, method_getImplementation(m1), method_getTypeEncoding(m1));
        }
        else{
            method_exchangeImplementations(m1, m2);
        }
    });
}

+ (void)load {
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        SEL s1 = @selector(go);
        SEL s2 = @selector(stop);
        Class class = object_getClass((id)self);
        Method m1 = class_getClassMethod(class, s1);
        Method m2 = class_getClassMethod(class, s2);
        method_exchangeImplementations(m1, m2);
}

第一种多了个判断,第二种是直接交换。

文章一开始也说了,第一种是比较严谨的写法;而第二种,当我们想交换有多个继承关系的子类里面的方法并且子类没有实现父类的方法时,直接method_exchangeImplementations会把父类的方法也给交换了,一般这不是我们想要的结果,下面我们直接撸代码来验证下:

父类 Person

.h

@interface Person : NSObject
//静态(类) 方法
+ (void)go;
+ (void)stop;
@end

.m

@implementation Person

+ (void)go {
    NSLog(@"Go!");
}

+ (void)stop {
    NSLog(@"Stop!");
}
@end

子类 Programmer

.h

#import "Person.h"

@interface Programmer : Person

@end

.m

#import "Programmer.h"
#import <objc/runtime.h>

@implementation Programmer

+ (void)load {
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        SEL s1 = @selector(go);
        SEL s2 = @selector(stop);
        Class class = object_getClass((id)self);
        Method m1 = class_getClassMethod(class, s1);
        Method m2 = class_getClassMethod(class, s2);
        method_exchangeImplementations(m1, m2);
    });
}


//+ (void)go {
//    NSLog(@"Programmer - go");
//}
//
//+ (void)stop {
//    NSLog(@"Programmer - stop");
//}
@end

调用

- (void)viewDidLoad {
    [super viewDidLoad];
    
    [Person go];
    [Person stop];
    NSLog(@"---------");
    [Programmer go];
    [Programmer stop];
    
}

首先,我们只实现父类的 gostop 这两个静态方法,打印如下:

2017-01-25 23:30:12.850 RuntimeDemo[4180:180408] Stop!
2017-01-25 23:30:12.851 RuntimeDemo[4180:180408] Go!
2017-01-25 23:30:12.851 RuntimeDemo[4180:180408] ---------
2017-01-25 23:30:12.851 RuntimeDemo[4180:180408] Stop!
2017-01-25 23:30:12.852 RuntimeDemo[4180:180408] Go!

可以发现,子类Programmer没有实现父类的方法直接交换时, 父类Person的方法也被交换了!

我们接着打开子类的 gostop 这两个静态方法,打印如下:

2017-01-25 23:32:22.953 RuntimeDemo[4215:181987] Go!
2017-01-25 23:32:22.954 RuntimeDemo[4215:181987] Stop!
2017-01-25 23:32:22.956 RuntimeDemo[4215:181987] ---------
2017-01-25 23:32:22.956 RuntimeDemo[4215:181987] Programmer - stop
2017-01-25 23:32:22.956 RuntimeDemo[4215:181987] Programmer - go

可以发现,子类实现了父类的方法,直接交换也是没有问题的!

我们加上判断:

+ (void)load {
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        ...
        
        BOOL success =  class_addMethod(class, s1, method_getImplementation(m2), method_getTypeEncoding(m2));
        if (success){
            class_replaceMethod(class, s2, method_getImplementation(m1), method_getTypeEncoding(m1));
        }else{
            method_exchangeImplementations(m1, m2);
        }
        
        ...
    });
}

把上面两种情况在试验下,可以发现交换的仅仅是子类Programmer的方法,父类Person是没有被交换的!大家可以自己尝试下。

稍稍的解释下:

class_addMethod函数会检查方法有没有实现,如果已经实现会返回 NO ,也就是直接走method_exchangeImplementations方法;没有实现会先在当前类增加一个新的实现方法,再把目标类中的方法通过class_replaceMethod函数替换为旧有的实现;

结尾

以上就是我遇到的问题,希望对大家能有点帮助。最后也希望大家能够亲自动手敲一遍,加深下印象。

参考链接

http://ios.jobbole.com/91962/

http://ios.jobbole.com/81657/

http://blog.csdn.net/horkychen/article/details/8532087

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

推荐阅读更多精彩内容

  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,559评论 18 399
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,580评论 18 139
  • 转至元数据结尾创建: 董潇伟,最新修改于: 十二月 23, 2016 转至元数据起始第一章:isa和Class一....
    40c0490e5268阅读 1,678评论 0 9
  • 我们常常会听说 Objective-C 是一门动态语言,那么这个「动态」表现在哪呢?我想最主要的表现就是 Obje...
    Ethan_Struggle阅读 2,165评论 0 7
  • 跟tab相反的快捷键:shift + tab; PS快捷键:取消快速选择:【Ctrl】+【D】钢笔工具描边时要活用...
    前端混合开发阅读 237评论 0 0