method swizzing

原理

类的方法类别中,选择子的名称通过映射表找到应该调用的方法。如下所示:
OC 的运行时提供了几个方法可以操作这张表。可以向其中新增选择子,改变选择子的实现,或者交换选择子映射到的指针。

在上图中,交换了 lowercaseStringuppercaseString 方法实现,新增了一个选择子 newSelector,修改了 capitalizedString 的实现。

方法交换实现 Demo

为 NSString 新增一个分类,交换 lowercaseString 方法

- (NSString*)my_lowerCase
{
    //交换后,my_lowerCase 调用的是系统的方法,不会循环调用
    NSString *lowerString = [self my_lowerCase];
    NSLog(@"called swapped lower String");
    return lowerString;
}  

load 方法里,进行方法调换

//交换实例方法
Method raw =  class_getInstanceMethod([NSString class], @selector(lowercaseString));
Method swap = class_getInstanceMethod([NSString class], @selector(my_lowerCase));
method_exchangeImplementations(raw, swap);

调用例子

NSString *testStr = @"MY";
NSString *lower = testStr.lowercaseString;

输出如下:

2017-12-27 21:28:40.661942+0800 MethodSwizzingDemo[96992:7927796] called swapped lower String

同样可以交换类方法。这个技术一般用于调试。

Method 和 IMP

runtime 的头文件里,看到如下:

/// An opaque type that represents a method in a class definition.
typedef struct objc_method *Method;

struct objc_method {
    SEL _Nonnull method_name                                 OBJC2_UNAVAILABLE;
    char * _Nullable method_types                            OBJC2_UNAVAILABLE;
    IMP _Nonnull method_imp                                  OBJC2_UNAVAILABLE;
}                                                            OBJC2_UNAVAILABLE;

即 IMP 是 Method 的一个属性。

posted on 2017-12-27 21:40  花老🐯  阅读(206)  评论(0编辑  收藏  举报

导航