用方法调配技术调试黑盒方法

Object-C是允许进行方法调配的,什么叫方法调配呢?解释一下。
以NSString类为例,其中有lowercaseString和uppercaseString方法,获取两个方法的实例:

Method originalMethod = class_getInstanceMethod([NSString class], @selector(lowercaseString));
Method swappedMethod = class_getInstanceMethod([NSString class], @selector(uppercaseString));

方法交换:

method_exchangeImplementations(originalMethod, swappedMethod);

从现在开始,如果在NSString实例上调用lowercaseString,那么执行的将是uppercaseString方法,反之亦然。

类的方法列表会把选择子的名称映射到相关的方法实现之上,使得“动态消息派发系统”能够据此找到应该调用的方法。这些方法均以函数指针的形式来表示,这种指针叫做IMP,其原型如下:

id (*IMP) (id, SEL, ...)

上述方法其实是交换了IMP指针。
但是在实际使用的过程中,交换两个方法的意义并不大,我们可以把这种技术应用在哪里呢?
我们可以在类别中写一个方法:

@implementation NSString (MyAdditions)
- (NSString *)myLowercaseString
{
    NSString *lowercase = [self myLowercaseString];
    NSLog(@"%@ => %@", self, lowercase);
    return lowercase;
}
@end

看上去此方法会死循环,不过此方法是准备和lowercaseString方法互换的:

Method originalMethod = class_getInstanceMethod([NSString class], @selector(lowercaseString));
Method swappedMethod = class_getInstanceMethod([NSString class], @selector(myLowercaseString));
method_exchangeImplementations(originalMethod, swappedMethod);

通过此方案,开发者可以为那些“完全不知道其具体实现”的黑盒方法增加日志记录功能,这非常有助于程序调试。然而只在调试时使用,很少有人采用此方法来永久改动某个类的功能。若是滥用,反而会令代码变得不易懂且难维护。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容