美文网首页
NSProxy实现伪多继承

NSProxy实现伪多继承

作者: seej | 来源:发表于2018-03-02 16:50 被阅读0次
    Objective-C是不支持多继承的,但是基于消息转发机制我们可以使用NSProxy来实现伪多继承。

    1.关于NSProxy

    NSProxy是和NSObject同级的一个类,它只实现了<NSObject>的协议。

    基于Objective-C的消息转发机制(iOS理解Objective-C中消息转发机制附Demo),在我们给NSProxy对象发送消息时,首先会根据selector在本类以及父类的方法列表中查找,如果找不到,则会启动消息转发机制。在一系列的消息转发过程中,如果 + resolveInstanceMethod: 以及 - forwardingTargetForSelector: 都未能对消息进行处理, - methodSignatureForSelector: 会被调用以获取方法签名,如果该方法返回了方法签名,则我们会在 - forwardInvocation: 方法中拿到相应消息的信息。


    2.实现

    首先我们需要创建一个继承自NSProxy的类并提供方法传入若干对象。

    + (instancetype)proxyWithObjs:(id)obj, ... NS_REQUIRES_NIL_TERMINATION {
        NSMutableArray * objs = [NSMutableArray arrayWithObject:obj];
        if (obj) {
            va_list args;
            va_start(args, obj);
            id obj;
            while ((obj = va_arg(args, id))) {
                [objs addObject:obj];
            }
            va_end(args);
        }
        SEEProxy * instance = [SEEProxy alloc];
        instance -> _objs = objs.copy;
        return instance;
    }
    

    当消息转发进行到 - methodSignatureForSelector: 时我们需要在当前拥有的对象中查找方法,找到后将方法签名返回,代码如下:

    - (NSMethodSignature *)methodSignatureForSelector:(SEL)sel {
        __block NSMethodSignature * signature;
        [_objs enumerateObjectsUsingBlock:^(id  _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
            //判断对象是否能够响应方法
            if ([obj respondsToSelector:sel]) {
                signature = [obj methodSignatureForSelector:sel];
                *stop = YES;
            }
        }];
        return signature;
    }
    

    返回方法签名后我们需要在 - forwardInvocation: 中对消息进行处理:

    - (void)forwardInvocation:(NSInvocation *)invocation {
        [_objs enumerateObjectsUsingBlock:^(id  _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
            //判断对象是否能够响应方法
            if ([obj respondsToSelector:invocation.selector]) {
                [invocation invokeWithTarget:obj];
                *stop = YES;
            }
        }];
    }
    

    我们需要通过 - performSelector: 调用方法,为了防止崩溃在调用前最好先使用 - respondsToSelector: 判断是否能够响应方法,因此我们还需要实现 - respondsToSelector: 方法:

    - (BOOL)respondsToSelector:(SEL)aSelector {
        __block BOOL flag = [super respondsToSelector:aSelector];
        if (flag) return flag;
        [_objs enumerateObjectsUsingBlock:^(id  _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
            flag = [obj respondsToSelector:aSelector];
            *stop = flag;
        }];
        return flag;
    }
    

    拓展:另外一种实现伪多继承的方式

    上文中我们是在消息转发过程中的第三步进行操作来实现伪多继承,纵观整个消息转发过程,我们在第二步同样有操作的空间,代码如下:

    - (id)forwardingTargetForSelector:(SEL)aSelector {
        __block id target;
        [_objs enumerateObjectsUsingBlock:^(id  _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
            //判断对象是否能够响应方法
            if ([obj respondsToSelector:aSelector]) {
                target = obj;
                *stop = YES;
            }
        }];
        return target;
    }
    

    以上为笔者对于伪多继承实现的一些见解,如文中有任何错误请及时指正。

    Demo地址

    参考

    iOS理解Objective-C中消息转发机制附Demo

    相关文章

      网友评论

          本文标题:NSProxy实现伪多继承

          本文链接:https://www.haomeiwen.com/subject/tcdaxftx.html