我正在编写UITableView的子类,我希望我的子类在将它们传递给“真正的”委托之前处理一些UITableViewDelegate方法,并转发我的子类未实现的所有UITableViewDelegate方法.
在子类中,我有一个私有财产:
@property (nonatomic,assign) id <UITableViewDelegate> trueDelegate;
它拥有所有未实现的方法应该转发的“真正的代表”.在我设置的两个init方法中
self.delegate = self;
我覆盖 – (void)setDelegate:(id)就像这样
-(void)setDelegate:(id<UITableViewDelegate>)delegate {
if (delegate != self) {
_trueDelegate = delegate;
} else {
[super setDelegate:self];
}
}
然后我重写这些来处理消息转发
-(NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector {
NSMethodSignature *sig;
sig = [[self.delegate class] instanceMethodSignatureForSelector:aSelector];
if (sig == nil) {
sig = [NSMethodSignature signatureWithObjCTypes:"@^v^c"];
}
return sig;
}
- (void)forwardInvocation:(NSInvocation *)anInvocation {
SEL selector = anInvocation.selector;
if ([self respondsToSelector:selector]) {
[anInvocation invokeWithTarget:self];
} else {
[anInvocation invokeWithTarget:_trueDelegate];
}
}
问题是未实现的委托方法永远不会在tableview上调用,因此它们没有机会被转发到_trueDelegate对象.
我试着在这里检查它们:
- (BOOL)respondsToSelector:(SEL)aSelector {
}
但是从来没有为UITableViewDelegate方法调用该方法,尽管它可以很好地捕获其他方法.
解决方法
为了提高性能,UITableView会在设置委托后立即检查并记住哪些委托方法可用.首先设置委托自我,然后设置trueDelegate.因此,当委托在UITableView上设置时,trueDelegate为nil,因此-respondsToSelector:on on always总是返回NO.
要解决此问题,请在设置trueDelegate后设置委托.此外,您可以简化转发代码.除去属性以外的所有代码,并将其替换为:
- (void)setDelegate:(id <UITableViewDelegate>)delegate
{
if (delegate == self) return;
self.trueDelegate = delegate;
[super setDelegate:self];
}
- (BOOL)respondsToSelector:(SEL)aSelector
{
if ([super respondsToSelector:aSelector]) return YES;
return [self.trueDelegate respondsToSelector:aSelector];
}
- (id)forwardingTargetForSelector:(SEL)aSelector
{
return self.trueDelegate;
}