前言
通过OC这个“调方法” = “向对象发消息”的机制,我们了解到方法在类中都是一张类似于字典的表。
selector即为字典的Key,IMP则为字典的值。通常发消息,都会在这张表中现寻找selector,然后再获取相应的具体实现(IMP)。
比如一个NSString类,其中的表大概如下:
Method Swizzling
Method Swizzling是runtime中的一个概念,指一个类的方法中,我们可以将selector相互对应的IMP进行调换。
比如我们将NSString类中的lowercaseString的实现方法与uppercaseString的实现方法做一个调换,结果如下:
以下用代码进行实现:
Method uppercaseMethod = class_getInstanceMethod([NSString class], @selector(uppercaseString));
Method lowercaseMethod = class_getInstanceMethod([NSString class], @selector(lowercaseString));
//class_getInstanceMethod方法主要是根据selector来获取方法的具体实现
method_exchangeImplementations(uppercaseMethod, lowercaseMethod);
//方法调换的主要方法,参数为方法的实现。
NSString *exampleString = @"AaBbCcDd";
NSLog(@"%@",[exampleString uppercaseString]);
NSLog(@"%@",[exampleString lowercaseString]);
后台输出如下:
2016-05-09 17:06:01.792 RunTimePlayGround[37295:5698418] aabbccdd
2016-05-09 17:06:01.795 RunTimePlayGround[37295:5698418] AABBCCDD
可以从两个打印看到,两个方法的实现已经跟selector完全相反了,这就是Method Swizzling的作用。
Debug
但是实际上,我们不应该随便的使用Method Swizzling,因为实在没有必要去交换两个方法的实现,这种情况少之又少。
不过我们可以利用Method Swizzling进行一些Debug操作。
打个比方,如果我们想在每次调用uppercaseString时候,后台都进行一次Log输出。
那我们第一个想到的方法可能是新建分类再重写一个方法或者单独写一个方法来专门进行Debug。
这样在实际运用之前我们可以这样做,但是当项目进行到一半,工程中已经有很多uppercaseString调用时,我们显然不希望再去大量修改import的类或者方法名的修改。
此时Method Swizzling就派上了用场。
我们可以写一个分类,准备一个用来与原声uppercaseString进行交换的方法。
比如在NSString+MethodSwizzling.h中
#import <Foundation/Foundation.h>
@interface NSString (MethodSwizzling)
- (NSString *)logUppercaseString;
@end
声明一个方法用于外部访问获取IMP。
在NSString+MethodSwizzling.m中:
#import "NSString+MethodSwizzling.h"
@implementation NSString (MethodSwizzling)
-(NSString *)logUppercaseString{
NSString *uppercaseString = [self logUppercaseString];
NSLog(@"used uppercaseString Method");
return uppercaseString;
}
@end
对声明方法进行实现。
以上方法看似是一个递归,但是等到实际运行我们实现过Method Swizzling后, 这里的NSString uppercaseString = [self logUppercaseString];
实际执行的是[self uppercaseString];*
测试代码(记得引入一次分类的头文件):
Method uppercaseMethod = class_getInstanceMethod([NSString class], @selector(uppercaseString));
Method lowercaseMethod = class_getInstanceMethod([NSString class], @selector(logUppercaseString));
//class_getInstanceMethod方法主要是根据selector来获取方法的具体实现
method_exchangeImplementations(uppercaseMethod, lowercaseMethod);
//方法调换的主要方法,参数为方法的实现。
NSString *exampleString = @"AaBbCcDd";
NSLog(@"%@",[exampleString uppercaseString]);
打印输出:
2016-05-09 17:24:24.189 RunTimePlayGround[37383:5732256] used uppercaseString Method
2016-05-09 17:24:24.189 RunTimePlayGround[37383:5732256] AABBCCDD
成功实现了我们之前所需要的Debug功能。
总结
- 1 谨慎善用Method Swizzling,不要再没有必要的地方引入,否则带来代码混乱,难读