iOS开发专区运行时runtimeiOS进阶相关

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

2017-07-05  本文已影响894人  Rxiaobing

一、多代理,多继承

对于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]用于判断是否包含某个类方法。

上一篇下一篇

猜你喜欢

热点阅读