我正在编写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 { }
解决方法
为了提高性能,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; }