swizzle英文解释为:(用酒棒等)搅合;那么Method Swizzling可以翻译为”方法交换”
方法由两个部分组成。Selector相当于一个方法的id;IMP是方法的实现。这样分开的一个便利之处是selector和IMP之间的对应关系可以被改变。比如一个 IMP 可以有多个 selectors 指向它。
而 Method Swizzling 可以交换两个方法的实现。或许你会问“什么情况下会需要这个呢?”。我们先来看下Objective-C中,两种扩展class的途径。首先是 subclassing。你可以重写某个方法,调用父类的实现,这也意味着你必须使用这个subclass的实例,但如果继承了某个Cocoa class,而Cocoa又返回了原先的class(比如 NSArray)。这种情况下,你会想添加一个方法到NSArray,也就是使用Category。99%的情况下这是OK的,但如果你重写了某个方法,就没有机会再调用原先的实现了。
Method Swizzling 可以搞定这个问题。你可以重写某个方法而不用继承,同时还可以调用原先的实现。通常的做法是在category中添加一个方法(当然也可以是一个全新的class)。可以通过method_exchangeImplementations
这个运行时方法来交换实现。来看一个demo,这个demo演示了如何重写addObject:方法
来记录每一个新添加的对象。
NSArray+Swizzle.h文件代码
#import <Foundation/Foundation.h>
@interface NSArray (Swizzle)
-(id)myLastObject;
@end
NSArray+Swizzle.m文件代码
#import "NSArray+Swizzle.h"
@implementation NSArray (Swizzle)
-(id)myLastObject{
id ret=[self myLastObject];
NSLog(@"This is myLastObject of wflytoc");
return ret;
}
@end
main.m文件代码
#import <Foundation/Foundation.h>
#import "NSArray+Swizzle.h"
#import <objc/runtime.h>
int main(int argc, const char * argv[]) {
@autoreleasepool {
Method ori_Mthod=class_getInstanceMethod([NSArray class], @selector(lastObject));
Method my_Mthod=class_getInstanceMethod([NSArray class], @selector(myLastObject));
method_exchangeImplementations(ori_Mthod, my_Mthod);
NSArray *array=@[@"1",@"2",@"3"];
NSString *string=[array lastObject];
NSLog(@"last object is %@",string);
}
return 0;
}
结果: