OC交换两个不同类的方法

8 篇文章 0 订阅
6 篇文章 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(@"交换方法失败--添加方法失败");
    }
 }

这个方法仅仅是交换两个不同类的实例方法,当然交换两个不同类的类方法一样是可行的,我在这里就不赘述了,详细的可以看我这个开源代码地址

  • 2
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
Objective-C中的方法交换可以通过`method_exchangeImplementations`函数实现,它可以交换两个方法的实现。 下面是一个示例代码,展示了如何交换两个方法的实现: ```objc #import <objc/runtime.h> @implementation MyClass - (void)originalMethod { NSLog(@"Original method"); } - (void)swizzledMethod { NSLog(@"Swizzled method"); } @end int main(int argc, char * argv[]) { @autoreleasepool { MyClass *myClass = [[MyClass alloc] init]; [myClass originalMethod]; // Output: Original method [myClass swizzledMethod]; // Output: Swizzled method Method originalMethod = class_getInstanceMethod([MyClass class], @selector(originalMethod)); Method swizzledMethod = class_getInstanceMethod([MyClass class], @selector(swizzledMethod)); method_exchangeImplementations(originalMethod, swizzledMethod); [myClass originalMethod]; // Output: Swizzled method [myClass swizzledMethod]; // Output: Original method } return 0; } ``` 在这个例子中,我们定义了一个`MyClass`,其中包含两个方法`originalMethod`和`swizzledMethod`。然后,我们使用`class_getInstanceMethod`函数获取两个方法的实现,使用`method_exchangeImplementations`函数交换它们的实现。最后,我们再次调用这两个方法,发现它们的实现已经被交换了。 需要注意的是,方法交换可能会影响到其他地方的代码,因此需要谨慎使用。在实际开发中,建议尽量避免使用方法交换,除非确实需要这样做。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值