序言
不知你是否在工作或者自己的代码中遇到过这样的情况?知道一个类的方法,并且自己想要自定义一个方法与其进行交换,但是呢,这个类是一个咋们看不到实现的类,更甚者是连头文件都看不到,尤其是面对系统自带界面里的某个view或者类。
正文
当然,开发至今,我也是很多时候遇到过类似的情况,并且是多见于修改系统view或者是某个连头文件都没有的类。可能你一开始会有这样的想法,写一个这个类的分类,然鹅,令人头疼的是,如果一开始就不知道这个类的定义的话,(也就是@interface XXX : NSObject),你即使自己自定义了这个分类,你也会过不了编译器那关,直接不让你编译通过。
当然,我也这么做过,这样失败之后,我接下来想到的就是,在自己写的一个类里面去自定义一个方法,之后在+load方法中去把自定义的方法跟想要交换的类的方法进行交换。当然,由于OC的语言特性以及它提供的强大的runtime函数,是肯定可以交换这两个不同类的方法的。
但是呢,痛点就来了,当你以为一切ok的时候,crash就静悄悄的走进了你的眼里。如果您对OC这门语言的底层不是很了解的话,您可能就会就此卡住,怎么都走不下了,甚至是为何会crash,都可能不知道,更别说解决这个crash了。
我这里就不对这个crash做详细的解释了,如果想真正搞明白为何crash,可以看我写的另一篇文章,OC消息发送机制完整全过程。我在这儿大概解释下这个crash的原因,在这里crash,是因为调进你方法里传过来的self参数不是你当前类的实例对象,而是被你交换的那个类的实例对象,因此,你重新想要通过调用你自定义的方法去调回原来的那个方法时,会直接给你报一个找不到该方法的错误,最终导致crash。
真正的原因就在于,原生方法的那个类中压根没有你自定义的那个方法的名字,因此,找不到并且crash就合情合理了。
那么,知道了这里,我就想到了一个方法,那就是,解决掉他这个找不到方法的问题,就是在交换的时候,将自定义的方法动态添加进原生方法的那个类中(感谢OC强大的runtime库啊!!!)。
P.S. 无论是交换类方法,还是实例方法,通过这个函数,你都只需传入类对象即可,即是调用某个类或者对象的class方法,或者是通过NSClassFromString(@"ClassString")函数获取到的值传入即可
通过这个函数,你可以在自定义的任何类里面自定义需要拿去交换的方法,之后将自定义的方法以及该方法在的那个类对象传过去即可.
具体代码如下:
/**
无论是交换类方法,还是实例方法,通过这个函数,你都只需传入类对象即可,即是调用某个类或者对象的class方法,或者是通过NSClassFromString(@"ClassString")函数获取到的值传入即可
通过这个函数,你可以在自定义的任何类里面自定义需要拿去交换的方法,之后将自定义的方法以及该方法在的那个类对象传过去即可
*/
/**
@brief 交换不同类中两个 对象方法 友好提示:自定义的方法可以写在任何的自定义类中 )
@param originalCls 被交换的类
@param swizzledCls 用来交换的类
@param originalSelector 被交换的方法
@param swizzledSelector 用来交换的方法
*/
void ZLSwizzleDifferentClassInstanceMethod(Class originalCls,Class swizzledCls,SEL originalSelector, SEL swizzledSelector) {
if (!originalCls || !swizzledCls) {
NSLog(@"交换方法失败--请保证交换的类名不为空");
return;
}
if (!originalSelector || !swizzledSelector) {
NSLog(@"交换方法失败--请保证交换的方法名不为空");
return;
}
Method originalMethod = class_getInstanceMethod(originalCls, originalSelector);
Method swizzledMethod = class_getInstanceMethod(swizzledCls, swizzledSelector);
BOOL didAddMethod = class_addMethod(originalCls,
swizzledSelector,
method_getImplementation(swizzledMethod),
method_getTypeEncoding(swizzledMethod));
if (didAddMethod) {
method_exchangeImplementations(originalMethod, class_getInstanceMethod(originalCls, swizzledSelector));
} else {
NSLog(@"交换方法失败--添加方法失败");
}
}
这个方法仅仅是交换两个不同类的实例方法,当然交换两个不同类的类方法一样是可行的,我在这里就不赘述了,详细的可以看我这个demo