美文网首页iOS开发专区运行时runtimeiOS进阶相关
ios之“多代理”,“多继承”,以及动态调用类方法实例方法

ios之“多代理”,“多继承”,以及动态调用类方法实例方法

作者: Rxiaobing | 来源:发表于2017-07-05 09:56 被阅读894次

一、多代理,多继承

对于Objective-C来说是不支持多继承的,由于消息机制名字查找发生在运行时而非编译时,很难解决多个基类可能导致的二义性问题。
不过其实 Objective-C 也无需支持多继承,我们可以找到如下几种间接实现多继承目的的方法:

消息转发
delegate和protocol
类别

消息转发

当向某个Object发送某消息,但runtime system在当前类和父类中都找不到对应方法的实现时,runtime system并不会立即报错使程序崩溃,而是依次执行下列步骤


1.从图中我们可以很明显的看出,当某个类没有实现某个方法的时候,会先向当前类发送resolveInstanceMethod信号,检查是否动态向该类添加了方法,如果未实现,则转而进入快速转发步骤;
2.向当前类发送forwardingTargetForSelector(快速转发)信号,若该方法返回值对象非nil或非self,检查返回的target是否实现了该方法,若实现了则调用这个方法,若为nil或者self,则会转而进行标准转发步骤;
3.如果未实现快速转发,则会向当前类发送methodSignatureForSelector:消息获取Selector对应的方法签名。返回值非空则通过forwardInvocation:转发消息,返回值为空则向当前对象发送doesNotRecognizeSelector:消息,程序崩溃退出(标准消息转发)

快速消息转发示例

    //Teacher类需要实现将消息转发给Doctor:
   - (id)forwardingTargetForSelector:(SEL)aSelector
  {
      Doctor *doctor = [[Doctor alloc]init];
  if ([doctor respondsToSelector:aSelector]) {
    return doctor;
  }
  return nil;
 } 

标准消息转发示例

- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector
{
    NSMethodSignature* signature = [super methodSignatureForSelector:aSelector];
    if (signature==nil) {
        for (id target  in self.targets){
            signature = [target methodSignatureForSelector:aSelector];
        }
        
    }
//    NSUInteger argCount = [signature numberOfArguments];//
//    for (NSInteger i=0 ; i<argCount ; i++) {//
//    }//
    
    return signature;
}

- (void)forwardInvocation:(NSInvocation *)anInvocation
{
    SEL seletor = [anInvocation selector];
    for (id target in self.targets) {
        if ([target respondsToSelector:seletor]) {
            [anInvocation invokeWithTarget:target];
        }

    }
    
}

两种消息转发方式的比较

快速消息转发:简单、快速、但仅能转发给一个对象
标准消息转发:稍复杂、较慢、但转发操作实现可控,可以实现多对象转发

在ios中的使用场景

场景一
场景一是取自开发文档的代码示例,他其实是通过消息传递机制实现了多重继承功能,使proxy拥有了NSSting与NSArray俩个类的方法属性
场景二
场景二是解决了NSTimer轮播功能的循环引用的问题。

二、动态调用类方法以及实例方法

在做项目的代码重构时,使用了mvvm模式与mvp模式相结合,然后就尽可能的抽取代码,尽量减少相同的代码,比如创建不同的cell,在我重建的项目中,所有plain类型的列表都共用了一个基类的tableviewcontroller,在子类中完全看不见和tableview相关的代码,然后在子类中根据不同的cell,创建带有不同identifier的cell!这个想法很快得到了实现,但是在动态创建cell的时候,却遇到了一个大问题,就是我无法根据(nsclassfromstring)生成的class来动态调用创建cell的方法,后来就使用了先regisiter tableviewCell类的方式!直至最近看了雷纯锋大神的mvvmreactcocoa以及afn中关于动态调用的一些方法,才发现如此简单,这里我也总结一下,希望能对各位朋友有所帮助,也是对我自己的几个总结与记忆

//创建cell
+ (id)cellWithTableView:(UITableView *)tableView identifier:(NSString *)identifier
{
    UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:identifier];
    if (!cell) {
        cell = [[self alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:identifier];
    }
    return cell;
}


Class cls = NSClassFromString(@"TableViewCell");
//虽然xcode不会提示,但是可以这样调用,为了安全在调用之前最好先cls instancerespondtoselector判断以下,以预防由于方法不存在导致app崩溃
    TableViewCell *cell = [cls cellWithTableView:tableView  identifier:@"dasd"];
    TableViewCell *cell1 = [cls performSelector:@selector(cellWithTableView:identifier:) withObject:tableView withObject:@"ceshi"];
    NSLog(@"%@  ===== %@",cell, cell1);

三、instancesRespondToSelector与respondsToSelector的区别

- (void)objectFun
{
    NSLog(@"object function");
}

+ (void)classFun
{
    NSLog(@"class function");
}


BOOL flag;

flag = [Test1 instancesRespondToSelector:@selector(objectFun)]; //YES

flag = [Test1 instancesRespondToSelector:@selector(classFun)]; //NO

flag = [Test1 respondsToSelector:@selector(objectFun)]; //NO

flag = [Test1 respondsToSelector:@selector(classFun)]; //YES

Test1 *obj = [[Test1 alloc] init];

flag = [obj respondsToSelector:@selector(objectFun)]; //YES

flag = [obj respondsToSelector:@selector(classFun)]; //NO

总结

  1. instancesRespondToSelector只能写在类名后面,respondsToSelector可以写在类名和实例名后面。

  2. [类 instancesRespondToSelector]判断的是该类的实例是否包含某方法,等效于:[该类的实例 respondsToSelector]。

  3. [类 respondsToSelector]用于判断是否包含某个类方法。

相关文章

  • ios之“多代理”,“多继承”,以及动态调用类方法实例方法

    一、多代理,多继承 对于Objective-C来说是不支持多继承的,由于消息机制名字查找发生在运行时而非编译时,很...

  • ES6解读3:类class

    类的继承 类的getter和setter方法 静态方法以及静态属性 注意:静态方法只能是类调用,不能实例调用

  • iOS动态调用类方法

    iOS动态调用类方法(不带参数) iOS动态调用类方法(带参数) iOS动态调用类方法(有返回值) 优点 弱化连接...

  • 动态代理

    动态代理实现过程 定义接口 获取代理实例并调用实力上的方法 动态代理会生成接口或者实现接口的类的实例 该实例并非静...

  • python-面向对象——继承

    继承 可以多层继承,子类可以直接调用父类的父类的函数 重写 调用被重写的方法: 多继承 python支持多继承 所...

  • 继承

    一. 单继承 继承的概念一般子女继承父辈 2.多继承 3.重写、调用父类方法 二. 静态方法和类方法 1、类方法 ...

  • Python基础语法(九)继承

    继承 Python 格式: 私有方法、属性不会被继承,所以子类不能调用父类的方法或属性 多继承 格式: 多继承扩展

  • python(进阶二)

    多态 封装 继承 mro 及多继承 实例与类之间的调用关系 property 魔法属性与方法 上下文管理器

  • 简述Python中多态,继承,封装,propertyy

    多态 封装 继承 mro 及多继承 实例与类之间的调用关系 property 魔法属性与方法 上下文管理器

  • python中一些高级语法和使用

    多态 封装 继承 mro 及多继承 实例与类之间的调用关系 property 魔法属性与方法 上下文管理器

网友评论

本文标题:ios之“多代理”,“多继承”,以及动态调用类方法实例方法

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