我想监控我的应用程序中的所有发布调用,因此我决定在 NSObject 上使用方法调配,以便可以监控所有类。实际代码取自此处,但为了清楚起见,我也会将其附加到此处。
似乎,尽管所有(大多数?)案例都有效,但这不适用于 UIView 及其子项。
我期待的是,即使 UIView 覆盖了这个方法,或者即使对它进行了调配,最终这个方法也会以原始的发布方法结束,因此通过调配我的代码将被执行。
如果我将 swizzling 放在 UIView 类和NSObject 类上,那么 swizzling 就完美了。如果我把它放在 NSOBject 上,Swizzling 在 UIResponder 类中也可以正常工作。似乎 UIView 有一些打破链条的自定义实现?
这是我正在使用的代码:
#import <objc/runtime.h>
#import <UIKit/UIKit.h>
@implementation NSObject (SwizzleRelease)
- (void)xxx_release {
printf("Will release %s #%d\n", [NSStringFromClass([self class]) UTF8String], [self retainCount]);
[self xxx_release];
}
+ (void)load {
NSLog(@"LOADED");
static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{
Class class = [self class];
SEL originalSelector = @selector(release);
SEL swizzledSelector = @selector(xxx_release);
Method originalMethod = class_getInstanceMethod(class, originalSelector);
Method swizzledMethod = class_getInstanceMethod(class, swizzledSelector);
BOOL didAddMethod =
class_addMethod(class,
originalSelector,
method_getImplementation(swizzledMethod),
method_getTypeEncoding(swizzledMethod));
if (didAddMethod) {
class_replaceMethod(class,
swizzledSelector,
method_getImplementation(originalMethod),
method_getTypeEncoding(originalMethod));
} else {
method_exchangeImplementations(originalMethod, swizzledMethod);
}
});
}
@end