发布网友 发布时间:2023-09-20 15:06
共1个回答
热心网友 时间:2024-12-05 01:47
�最底层的转发函数为 objc_msgSend ,它的定义如下
从以上的定义我们可以得出一个消息转发包含了几大要素:target、selector、arguments、return value, objc_msgSend 是 C 函数,苹果不提倡我们直接使用该函数来向对象消息。
想必大家都知道使用 performSelector 给对象发送消息,但是其有几个短板
NSInvocation 是苹果工程师们提供的一个高层的消息转发系统。它是一个命令对象,可以给任何 Objective-C 对象类型发送消息,接下来将介绍 NSInvocation 的�用法。
必须使用工厂方法 invocationWithMethodSignature: 来创建一个 NSInvocation 实例。工厂方法的参数是一个 NSMethodSignature 对象。一般使用 NSObject 的实例方法 methodSignatureForSelector: 或者类方法 instanceMethodSignatureForSelector: 来创建对应 selector 的 NSMethodSignature 对象。
例:创建类方法的签名与实例方法签名
需要注意的是 NSMethodSignature 对象仅仅表示了方法的签名:方法的请求、返回数据的编码。所以在使用 NSMethodSignature 来创建 NSInvocation 对象之后仍需指定消息的接收对象和选择子。
原则是接收对象的对应选择子需要跟 NSMethodSignature 相匹配。但是根据实践来说,只要不造成 NSInvocation setArgument:atIndex 越界的异常,都是可以成功转发消息的,并且转发成功之后,未赋值的参数都将被赋值为 nil。
例如:
执行结果:
以上为 NSInvocation 类中定义针对参数的操作。 argumentLocation 参数为 void * 类型,表示需要传递指针地址给它。idx 参数是从 2 开始的, 0 和 1 分别代表 target 和 selector,虽然可以�直接使用 getArgument:atIndex 来获取 target 和 selector,但是不如 NSInvocation 的 target 以及 selector 属性来的方便 。需要注意的是当 idx 超过对应 NSMethodSignature的参数个数的时候获取参数和设置参数的方法都会抛出 NSInvalidArgumentException 异常。
例如:给 greetingWithName: 方法传参
需要特别注意 setArgument:atIndex: 默认不会强引用它的 argument,如果 argument 在 NSInvocation 执行的时候之前被释放就会造成野指针异常(EXC_BAD_ACCESS)。
如上图所示, invocation 未�强引用它的 target,在控制器弹出之后,target �被释放,然后再 invoke 这个 invocation 会造成野指针异常。调用 retainArguments 方法来强引用参数(包括 target 以及 selector)。
NSInvocation 类中的返回数据的方法如下
可以看到返回数据仍然是通过传入指针来进传值的。例:
输出结果为:
需要注意的是:考虑到 getReturnValue 方法仅仅是将返回数据拷贝到提供的缓存区(retLoc)内,并不会考虑到此处的 内存管理 ,所以如果返回数据是对象类型的,实际上获取到的返回数据是 __unsafe_unretained 类型的,上层函数再�把它作为返回数据返回的时候就会造成野指针异常。通常的解决方法有2种:
第一种:新建一个相同类型的对象并指向它,这样做 result 就会强引用 tempResult,当做返回数据返回之后会自动添加 autorelease 关键字,也就不会造成野指针异常。
第二种:�使用 __bridge 将缓存区转换为 Objective-C 类型,这种做法其实跟第一种相似,但是我们更建议使用这种方式来解决以上问题,因为 getReturnValue �本来就是给缓存区写入数据,缓存区声明为 void* 类型更为合理,然后通过 __bridge 方式转换为 Objective-C 类型并�且将该内存区的内存管理交给 ARC。