[iOS] Method-Swizzling方法交换

1. Method-Swizzling

1.1 简介
  • Runtime 中的黑魔法,运行时替换方法的实现
  • OC 中利用 Method-Swizzling实现 AOP(面向切片编程)
  • 每个方法Method 中都有 SELIMP,方法交换,就是将SELIMP 的对应关系断开,将SEL 和新的IMP 建立关系

如下图所示:


image.jpeg
1.2 相关的 API
// 通过 sel 获取实例方法
class_getInstanceMethod

// 通过 sel 获取类方法
class_getClassMethod

// 获取一个方法的实现
method_getImplementation

// 设置一个方法的实现
method_setImplementation

// 获取方法实现的编码类型
method_getTypeEncoding

// 添加方法实现
class_addMethod

// 用一个方法的实现,替换另一个方法的实现,并不是交换
class_replaceMethod:

// 交换两个方法的实现
method_exchangeImplementations

2. 问题记录

2.1 method-swizzling使用过程中的一次性问题

如果写在+load 方法中会调用多次,这样会导致方法的重复交换,使 sel 的指向又恢复成原来的imp,可以使用 dispatch_once 实现只调用一次:

+ (void)load{
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        [LGRuntimeTool lg_bestMethodSwizzlingWithClass:self oriSEL:@selector(helloword) swizzledSEL:@selector(lg_studentInstanceMethod)];
    });
}
2.2 子类中替换从父类继承的方法

在下面这段代码中,Person 实现了personInstanceMethod,而 Teahcer 继承自Person,但没有实现personInstanceMethod,并且将personInstanceMethod替换成了自己的方法实现,看下面这段代码会有什么问题?

/////////////////////// Person类:
@interface Person : NSObject

- (void)personInstanceMethod;

@end
@implementation Person

- (void)personInstanceMethod{
    NSLog(@"person 对象方法:%s",__func__);
}

@end

//////////////////////// Teacher类

@interface Teacher : Person

@end

@implementation Teacher

+ (void)load{
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        
        // 交换方法
        Method oriMethod = class_getInstanceMethod(self, @selector(personInstanceMethod));
        Method swiMethod = class_getInstanceMethod(self, @selector(lg_teacherInstanceMethod));
        method_exchangeImplementations(oriMethod, swiMethod);
        
    });
}

// 这里是一个注意的点,这里并不是递归调用,因为已经交换完毕了,lg_teacherInstanceMethod会调用到 oriIMP,即 personInstanceMethod 的方法实现
- (void) lg_teacherInstanceMethod{
    [self lg_teacherInstanceMethod];
    NSLog(@"Teahcer分类添加的lg对象方法:%s",__func__);
}

@end

////////////// 调用:

- (void)viewDidLoad {
    [super viewDidLoad];
    
    Person *person = [Person alloc];
    [person personInstanceMethod];
    
    Teacher *teacher = [Teacher alloc];
    [teacher personInstanceMethod];

}

直接运行代码,会发现 person 调用personInstanceMethod方法时产生崩溃:

截屏2021-01-15 上午11.00.32.png

personInstanceMethod的方法实现在 Teacher类中被替换成了lg_teacherInstanceMethod的方法实现,但是这个方法实现是写在 Teacher类中的,在 Person类中并没有这个方法实现,所以当调用时找不到相关的imp,产生崩溃。

我们这样替换了父类的方法,影响到了父类,所以正确的做法是先将oriSEL 方法尝试添加到 Teacher 类中,如果Teacher类有这个方法(不是从父类继承的),就不添加,直接exchange,否则给Teacher类添加oriSEL方法,再将swiSEL的实现设置成oriSEL的实现,这样不会影响其父类

这里将替换方法抽取出来,如下:

+ (void)methodSwizzlingWithClass:(Class)cls oriSEL:(SEL)oriSEL swizzledSEL:(SEL)swizzledSEL{
    
    if (!cls) NSLog(@"传入的交换类不能为空");
    
    Method oriMethod = class_getInstanceMethod(cls, oriSEL);
    Method swiMethod = class_getInstanceMethod(cls, swizzledSEL);
   
    // class_addMethod方法内部会判断当前类是否有 oriSEL 方法(不是从父类继承的),如果没有则会添加 oriSEL 方法 ,方法实现为swiMethod
    BOOL success = class_addMethod(cls, oriSEL, method_getImplementation(swiMethod), method_getTypeEncoding(swiMethod));

    if (success) {// 如果没有,添加成功后就进行方法替换, 将 swiSEL 方法的实现替换成 oriSEL 方法的实现
        class_replaceMethod(cls, swizzledSEL, method_getImplementation(oriMethod), method_getTypeEncoding(oriMethod));
    }else{ // 添加失败,说明当前类有这个方法,直接交换
        method_exchangeImplementations(oriMethod, swiMethod);
    }   
}
  • 下面是class_replaceMethodclass_addMethodmethod_exchangeImplementations的源码实现:
image.jpeg
  • 其中class_replaceMethodclass_addMethod中都调用了 addMethod方法,区别在于最后replace 值的判断,下面是 addMethod的实现:
    image.jpeg

注意:getMethodNoSuper_nolock 是判断自己类是否有这个方法,不会从父类中查找判断。

2.3 子类中没有实现,父类也没有实现

在调用personInstanceMethod方法时,父类Person中只有声明,没有实现,子类Teacher中既没有声明,也没有实现,

/////////////////////// Person类:
@interface Person : NSObject

- (void)personInstanceMethod;

@end
@implementation Person


@end

//////////////////////// Teacher类

@interface Teacher : Person

@end

@implementation Teacher

@end

经过调试,发现运行代码会崩溃,报错结果如下所示:


截屏2021-01-16 下午7.13.33.png

原因是 personInstanceMethod没有实现,当给 lg_teacherInstanceMethod设置oriMethod 的方法实现时,由于 oriMethod 的 imp 为空,所以设置失败,导致lg_teacherInstanceMethod会一直调用自己:

截屏2021-01-16 下午7.31.09.png

如果oriMethod 为空,为了崩溃需要额外加一层判断:

  • 通过 class_addMethodoriSEL添加swiMethod 方法实现
  • 通过 method_setImplementationswiMethodimp 指向不做任何事的空实现
+ (void)exchangeImpWithClass:(Class)class oriSEL:(SEL)oriSEL swiSEL:(SEL)swiSEL{
    Method oriMethod = class_getInstanceMethod(class, oriSEL);
    Method swiMethod = class_getInstanceMethod(class, swiSEL);
    
    if(!oriMethod){
        class_addMethod(class, oriSEL, method_getImplementation(swiMethod), method_getTypeEncoding(oriMethod));
        method_setImplementation(swiMethod, imp_implementationWithBlock(^(id self, SEL _cmd){
            NSLog(@"来了一个空的 imp");
        }));
    }
    
    BOOL success = class_addMethod(class, oriSEL, method_getImplementation(swiMethod), method_getTypeEncoding(swiMethod));
    if(success){
        class_replaceMethod(class, swiSEL, method_getImplementation(oriMethod), method_getTypeEncoding(oriMethod));
    }else{
        method_exchangeImplementations(oriMethod, swiMethod);
    }
}

3. 使用场景

一般用的挺多的是防止数组、字典等越界崩溃,或者添加的value 值为 nil,有一个注意的点:在iOS中NSNumber、NSArray、NSDictionary等这些类都是类簇,一个NSArray的实现可能由多个类组成。所以如果想对NSArray进行Swizzling,必须获取到其“真身”进行Swizzling,直接对NSArray进行操作是无效的。
下面列举了NSArrayNSDictionary本类的类名,可以通过Runtime函数取出本类。

真身
NSArray __NSArrayI
NSMutableArray __NSArrayM
NSDictionary __NSDictionaryI
NSMutableDictionary __NSDictionaryM

比如替换NSArray 的方法,需要获取类名为__NSArrayI的类:

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

推荐阅读更多精彩内容