消息转发与NSProxy

消息转发

消息转发与NSProxy_第1张图片
消息转发步骤

  1. 调用resolveInstanceMethod:方法 (或 resolveClassMethod:)。允许用户在此时为该 Class 动态添加实现。如果有实现了,则调用并返回YES,那么重新开始objc_msgSend流程。这一次对象会响应这个选择器,一般是因为它已经调用过 class_addMethod。如果仍没实现,继续下面的动作。
  2. 调用forwardingTargetForSelector:方法,尝试找到一个能响应该消息的对象。如果获取到,则直接把消息转发给它,返回非 nil 对象。否则返回 nil ,继续下面的动作。注意,这里不要返回 self ,否则会形成死循环。
  3. 调用methodSignatureForSelector:方法,尝试获得一个方法签名。如果获取不到,则直接调用doesNotRecognizeSelector抛出异常。如果能获取,则返回非nil:创建一个 NSlnvocation 并传给forwardInvocation:。
  4. 调用forwardInvocation:方法,将第3步获取到的方法签名包装成 Invocation 传入,如何处理就在这里面了,并返回非ni。
  5. 调用doesNotRecognizeSelector: ,默认的实现是抛出异常。如果第3步没能获得一个方法签名,执行该步骤。
    上面前4个方法均是模板方法,开发者可以override,由 runtime 来调用。最常见的实现消息转发:就是重写方法3和4,吞掉一个消息或者代理给其他对象都是没问题的

也就是说_objc_msgForward在进行消息转发的过程中会涉及以下这几个方法:

resolveInstanceMethod:方法 (或 resolveClassMethod:)。
forwardingTargetForSelector:方法
methodSignatureForSelector:方法
forwardInvocation:方法
doesNotRecognizeSelector: 方法

NSProxy

概述

NSProxy是一个为对象定义接口的抽象父类,并且为其它对象或者一些不存在的对象扮演了替身角色。通常,给proxy的消息被转发给实际对象或者导致proxy加载(转化它为)实际对象。NSProxy的子类能被用来实现透明的分布式消息(例如:NSDistantObject)或者延缓要花费昂贵代价创建的对象的实现。

主要是利用消息转发机制

声明

消息转发与NSProxy_第2张图片

没有init方法,直接alloc。并且可以看到,它遵守了 NSObject 协议,并且第一个 Ivar 是一个 isa 指针,因此它完全是可以拿来当一个 NSObject 或其派生类来使用的。

一般使用两个方法

// 生成方法签名
- (nullable NSMethodSignature *)methodSignatureForSelector:(SEL)sel 
// 方法的签名配发
- (void)forwardInvocation:(NSInvocation *)invocation 

三个例子

例子一

取自开发文档的代码示例,他其实是通过消息传递机制实现了多重继承功能,使proxy拥有了NSSting与NSArray俩个类的方法属性

例子二

解决了NSTimer轮播功能的循环引用的问题。其实我以前没有太注意这个问题,以后可以引以为戒

例子三

就是如何实现偷天换日的功能,比如,如何让人像汽车一样run

#import 



@interface Person : NSObject

- (void)run;

@end


#import "Person.h"

#import "Car.h"



@implementation Person



//返回和参数的类型信息

- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector{

    NSString *sel = NSStringFromSelector(aSelector);

    if ([sel isEqualToString:@"run"]) {

        return [NSMethodSNSInvocationignature signatureWithObjCTypes:"v@:"];

    }

    return [super methodSignatureForSelector:aSelector];

    //引发NSInvalidArgumentException。在你具体子类中重写这个方法,为被给选择器和你的代理对象代表的类返回合适的NSMethodSignature对象

}

//加载对象,把对象传递给anInvocation

- (void)forwardInvocation:(NSInvocation *)anInvocation{

    SEL selector = [anInvocation selector];

    Car *car = [[Car alloc] init];

    if ([car respondsToSelector:selector]) {

        [anInvocation invokeWithTarget:car];//传递一个invocation给proxy代表的真的对象

    }

}


#import 



@interface Car : NSObject

- (void)run;

@end



#import "Car.h"



@implementation Car

- (void)run{

    NSLog(@"Car is running!");

}

@end

以上的例子都是通过methodSignatureForSelector:和forwardInvocation:实现消息转发来达到目的

你可能感兴趣的:(ios开发,runtime)