我正在编写UITableView的子类,我希望我的子类在将它们传递给“真实”委托之前,先处理一些UITableViewDelegate方法本身,并转发未由我的子类实现的所有UITableViewDelegate方法。

在子类中,我有一个私有财产:

@property (nonatomic, assign) id <UITableViewDelegate> trueDelegate;

其中包含所有未实现的方法应转发给的“真实委托”。在我的两种初始化方法中,
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];
    }
}

问题是未实现的委托方法永远不会在表视图上被调用,因此没有机会将它们转发到_trueDelegate对象。

我尝试在这里检查它们:
- (BOOL)respondsToSelector:(SEL)aSelector {

}

但是,尽管可以捕获其他方法,但从未为UITableViewDelegate方法调用该方法。

最佳答案

为了提高性能,UITableView检查并记住设置了委托后立即可用的委托方法。首先设置委托self,然后设置trueDelegate。因此,在将委托设置在UITableView上时,trueDelegatenil,因此该代表上的-respondsToSelector:始终返回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;
}

07-28 06:21