原理:
方法交换是基于OC的动态绑定技术实现的,动态绑定简单的说就是我们OC对象直到运行时才把这个对象所具有的方法和属性绑定到对象上.这些方法和属性有我们类中原先实现的,也有动态生成的,还有替换掉原来实现的,这样我们就可以在运行时动态的修改方法实现了.
大家应该知道我们对象的方法都在对象的isa指针所指的class的方法列表中,这个列表维护的一张selector名称和imp的映射表,那我们的Method Swizzling实际上就是篡改这个表的映射关系,从而达到方法交换的目的.
应用:
Method Swizzling 有着相当广泛的应用场景,因为这项技术不仅能交换我们自己编写的方法,还能交换系统方法,从而可以在系统方法中添加一些功能,或者直接全部修改实现.
举例说明
比如你一个正常迭代的项目,突然有个需求是在每个控制器vieDidLoad的时候增加个功能(比如打印log:没有实际意义,只是举例说明),那么怎么办?
1.新建个基类,然后让每个控制器都继承这个基类,苦逼的是要每个控制器都换住个修改
2.创建viewCotroller的分类,覆盖系统的viewDidLoad方法,但是这是分类的坑,不要去覆盖系统的方法,这样就只会走重写后分类的方法,不会走原方法,到时候debug可能头炸掉都找不到bug
3.方法交换,在分类里面用自己的方法和系统的方法交换,完成需求
方法交换的实现
我们先给UIViewController添加一个Category,然后在Category中的+(void)load方法中添加Method Swizzling方法,我们用来替换的方法也写在这个Category中。由于load类方法是程序运行时这个类被加载到内存中就调用的一个方法,执行比较早,并且不需要我们手动调用。而且这个方法具有唯一性,也就是只会被调用一次,不用担心资源抢夺
#import "UIViewController+Swizzling.h"
//方法交换被忘记导入runtime头文件
#import <objc/runtime.h>
@implementation UIViewController (Swizzling)
+ (void)load {
//获取系统类库的方法viewDidLoad
Method originalMethod = class_getInstanceMethod([self class], @selector(viewDidLoad));
//定义交换的方法
Method SwizzlingMethod = class_getInstanceMethod([self class], @selector(SwizllingViewDidLoad));
//交换
method_exchangeImplementations(originalMethod, SwizzlingMethod);
}
2.实现自己的方法
- (void)SwizllingViewDidLoad {
NSLog(@"成功");
//这地方并不会造成死循环,这一步也是方法交换的关键,此时调用本身,由于做了交换,其实是执行的viewDidLoad的方法
[self SwizllingViewDidLoad];
}
到此方法交换就完成了,其实并不难,看下运行的效果
我在viewController的viewDidLoad的方法里面什么代码也没有写
但是运行的结果是打印了"成功"两个字
下面图解一下方法交换运行后的步骤,应该更清晰一些
demo做的方法交换并没有实际意义,只是模拟方法交换的使用,更多的使用比如数组越界,数组添加nil的崩溃都可以用方法交换来拦截防止崩溃,等你熟悉了,其实很多地方都可以用来优化你的项目