美文网首页
iOS交换两个不同类的方法

iOS交换两个不同类的方法

作者: Invisible_He | 来源:发表于2019-12-24 23:28 被阅读0次

    序言

    不知你是否在工作或者自己的代码中遇到过这样的情况?知道一个类的方法,并且自己想要自定义一个方法与其进行交换,但是呢,这个类是一个咋们看不到实现的类,更甚者是连头文件都看不到,尤其是面对系统自带界面里的某个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

    相关文章

      网友评论

          本文标题:iOS交换两个不同类的方法

          本文链接:https://www.haomeiwen.com/subject/tuyfoctx.html