美文网首页
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实现伪多继承

    Objective-C是不支持多继承的,但是基于消息转发机制我们可以使用NSProxy来实现伪多继承。 1.关于N...

  • iOS NSProxy使用

    简介:通过NSProxy 可以实现类的"伪多继承",demo中KLProxy通过拦截方法修改了cat和dog本来的...

  • NSProxy 实现“多继承”

    之前有在YYWeakProxy看到过 NSProxy 使用,可以解决NSTimer 对 target 是强引用的问...

  • 协议和 NSProxy 实现多继承

    OC 中一个类只有一个父类, 这就是单一继承, 但是我们可以用协议和 NSProxy 实现多继承 先说协议, 协议...

  • NSProxy 使用总结

    NSProxy是一个虚类。它有什么用处呢?OC中类是不支持多继承的,要想实现多继承一般是有protocol的方式,...

  • iOS:NSProxy

    OC 中一个类只有一个父类,这就是单一继承,但是我们可以用协议和NSProxy实现多继承。 1、 protocol...

  • 记录NSObject 的关系

    NSObject的关系 NSProxy 没有继承NSObject,只实现了NSObject的协议,所以NSProx...

  • iOS中的“多继承”

    OC中是不支持多继承的,但是某些情况下想实现类似多继承的效果怎么办呢?简单的总结了几种实现伪多继承的方法。不对之处...

  • OC中如何实现”多继承“,(NSProxy,Protocol)

    OC 中一个类只有一个父类 但是我们可以用协议和 NSProxy 实现多继承 1. protocol 先说协议 比...

  • javaScript对象的继承

    目录: 伪继承 通过中间对象实现 ES6 class实现继承关系 1.伪继承 父类: 子类: 示例: 接着上面的代...

网友评论

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

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