iOS Runtime的消息转发机制
前面我们已经讲解Runtime的基本概念和基本使用,如果大家对Runtime机制不是很了解,可以先看一下以前的博客,会对理解这篇博客有所帮助!!!
Runtime基本概念:https://www.cnblogs.com/guohai-stronger/p/9154889.html
Runtime基本使用:https://www.cnblogs.com/guohai-stronger/p/9184356.html
看过上面两篇博客之后,大家可能会对Runtime有所了解,但在上述讲述Runtime还有一个问题,当对象无法接收到相关的消息时候,又会发生什么?这就是我们即将讲的消息转发机制。
一、前言
在讲述消息转发机制前,先通过一个简单的例子
id num = @;
//输出123
NSLog(@"%@", num);
//程序崩溃,报错[__NSCFNumber appendString:]: unrecognized selector sent to instance 0x7b27
[num appendString:@"Hello World"];
在程序代码中加入上面代码,运行过程为:先在相关的类中,搜索方法列表,如果出现找不到则会一直向上搜索到根部(也就是NSObject);如果出现还找不到并且消息转发都失败,此时就会执行doesNotRecognizeSelector:紧接着方法报unrecognized selector错误?
下面我们来讲述消息转发的过程中,为什么经常出现消息的最后三次机会?
第一步:所属类的动态方法解析
首先,如果沿着继承树没有搜索到相关的方法时,则就会向接受者所属的类进行一次请求,看是否可以动态的添加一个方法,如下:
+(BOOL)resolveInstanceMethod:(SEL)name
注意这是一个类方法,因为向接受者所属的类进行一次请求。
下面我们举个例子:
#import "ViewController.h"
#import <objc/runtime.h> @interface ViewController () @end @implementation ViewController - (void)viewDidLoad {
[super viewDidLoad];
self.title = @"Runtime";
[self performSelector:@selector(dynamicSelector) withObject:nil];
} + (BOOL)resolveInstanceMethod:(SEL)sel{
if (sel == @selector(dynamicSelector)) {
class_addMethod([self class],sel, (IMP)myMehtod,"v@:");
return YES;
}else{
return [super resolveInstanceMethod:sel];
} } void myMehtod(id self,SEL _cmd){
NSLog(@"This is added dynamic");
}
在看结果之前,我们先简单了解下:class_addMethod(上面出现的)
BOOL class_addMethod(Class cls, SEL name, IMP imp, const char *types)
此函数共有四个参数:
- Class cls:要添加的类对象
- name:添加后的selector方法名字
- imp:实现方法
- type:方法参数的编码
方法的返回值为BOOL类型:YES表示本类可以处理;NO代表需要使用转发机制。
下面是上面的运行结果,结果如下:
如果第一步不能处理,会进行到第二步。
第二步:调用forwardingTargetForSelector把任务转发给另一个对象。
#import "ViewController.h"
#import <objc/runtime.h> @interface CustomObject : NSObject
-(void)dynamicSelector;
@end
@implementation CustomObject -(void)dynamicSelector{
NSLog(@"hello world");
}
@end @interface ViewController ()
@property (strong,nonatomic)CustomObject * myObj;
@end
@implementation ViewController - (void)viewDidLoad {
[super viewDidLoad];
self.myObj = [[CustomObject alloc] init];
[self performSelector:@selector(dynamicSelector) withObject:nil]; } -(id)forwardingTargetForSelector:(SEL)aSelector{
if (aSelector == @selector(dynamicSelector) && [self.myObj respondsToSelector:@selector(dynamicSelector)]) {
return self.myObj;
}else{
return [super forwardingTargetForSelector:aSelector];
} }
@end
如果第二步骤也不能处理的时候,会交给第三步骤。
第三步:调用forwardInvocation转发给其他
#import "ViewController.h"
#import <objc/runtime.h> @interface CustomObject : NSObject
-(void)dynamicSelector;
@end
@implementation CustomObject -(void)dynamicSelector{
NSLog(@"hello world");
}
@end
@interface ViewController ()
@property (strong,nonatomic)CustomObject * myObj;
@end
@implementation ViewController - (void)viewDidLoad {
[super viewDidLoad];
self.myObj = [[CustomObject alloc] init];
[self performSelector:@selector(dynamicSelector) withObject:nil]; }
-(void)forwardInvocation:(NSInvocation *)anInvocation{
if ([self.myObj respondsToSelector:[anInvocation selector]]) {
[anInvocation invokeWithTarget:self.myObj];
}else{
[super forwardInvocation:anInvocation];
}
}
- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector {
return [CustomObject instanceMethodSignatureForSelector:aSelector];
}
@end
上面就是iOS Runtime消息的转发机制,欢迎指正!!!
iOS Runtime的消息转发机制的更多相关文章
- OC:浅析Runtime中消息转发机制
一.介绍 OC是一门动态性语言,其实现的本质是利用runtime机制.在runtime中,对象调用方法,其实就是给对象发送一个消息,也即objc_msgSend().在这个消息发送的过程中,系统会进行 ...
- iOS的消息转发机制详解
iOS开发过程中,有一类的错误会经常遇到,就是找不到所调用的方法,当然这类问题比较好解决,给当前对象或其父类对象添加该方法即可,使得编译器在编译时能正确找到该方法:或者,还有另外的方法,由于Objec ...
- iOS runtime探究(二): 从runtime開始深入理解OC消息转发机制
你要知道的runtime都在这里 转载请注明出处 http://blog.csdn.net/u014205968/article/details/67639289 本文主要解说runtime相关知识, ...
- iOS 消息转发机制
这篇博客的前置知识点是 OC 的消息传递机制,如果你对此还不了解,请先学习之,再来看这篇.这篇博客我尝试用口语的方式像讲述 PPT 一样给大家讲述这个知识点. 我们来思考一个问题,如果对象在收到无法解 ...
- iOS消息转发机制
iOS消息转发机制 “消息派发系统”(message-dispatch system) 若想令类能够理解某条消息,我们必须实现出对应的方法才行.但是,在编译器向类发送其无法解读的消息时并不会报错,因为 ...
- 理解Objective-C Runtime(三)消息转发机制
消息转发机制概述 上一篇博客消息传递机制中讲解了Objective-C中对象的「消息传递机制」.本文需要讲解另外一个重要问题:当对象受到无法处理的消息之后会发生什么情况? 显然,若想令类能理解某条消息 ...
- runtime之消息转发
前言 在上一篇文章中我们初尝了runtime的黑魔法,可以在程序编译阶段就获取到成员变量的名字,特性以及动态的给对象增加属性等等,在接下来中我们进一步了解OC的消息发送机制.如果之前没接触过runti ...
- Effective Objective-C 2.0 — 第12条:理解消息转发机制
11 条讲解了对象的消息传递机制 12条讲解对象在收到无法解读的消息之后会发生什么,就会启动“消息转发”(message forwarding)机制, 若对象无法响应某个选择子,则进入消息转发流程. ...
- runtime消息转发机制
Objective-C 扩展了 C 语言,并加入了面向对象特性和 Smalltalk 式的消息传递机制.而这个扩展的核心是一个用 C 和 编译语言 写的 Runtime 库.它是 Objective- ...
随机推荐
- dhcp服务简单配置
dhcp服务搭建 注意事项: > 配置虚拟机虚拟网络编辑器,取消"使用本地DHCP服务将IP地址分配给虚拟机" > 虚拟机网络连接设置为"仅主机模式" ...
- 扁平化promise调用链(译)
这是对Flattened Promise Chains的翻译,水平有限请见谅^ ^. Promises对于解决复杂异步请求与响应问题堪称伟大.AngularJS提供了$q和$http来实现它:还有很多 ...
- webservice的两种方式SOAP和REST的通俗理解
Webservice代表所有基于web的服务,包含两种方式SOAP和REST 以SOAP为例: 一个RPC call 就是把一个XML文档post到某个URL下,这个xml文档里写明我要调用的函数名和 ...
- rails应用的部署
简单部署 RAILS_ENV=production rake secret /etc/profile export SECRET_KEY_BASE=刚才生成的密钥 source /etc/profil ...
- SpringBoot 基础01
SpringBoot 基础 pom.xml <!-- Spring Boot 依赖版本控制 --> <parent> <groupId>org.springfram ...
- G++与C++的区别
C++是一门计算机编程语言,G++不是语言,是一款编译器中编译C++程序的命令而已. 不同的编译器,会对代码做出一些不同的优化 比如说: a++; 和 ++a; 如果从标准C的角度去理解.a++这个 ...
- 如何封装使用api形式调用的vue组件
在实际开发中一般有两种封装vue组件的方法:一种就是常用的的通过props父组件传值给子组件的方法: 子组件 父组件: 还有一种就是通过调用api的形式,下面例子是本人在实际项目中封装的一个自定义图标 ...
- iOS TouchID & FaceID
import UIKit import LocalAuthentication //指纹识别必须用真机测试,并且在iOS8以上系统,如果是FaceID至少IOS11以上. class Authenti ...
- day20_雷神_django第三天
django第三天 动态路由,实现统一删除功能 urls url(r'^host_del/(host|user|group)/(\d+)$',views.host_del,name='del'), t ...
- why microsoft named their cloud service Azure?
best guess I can personally make is that because Azure literally means “bright blue color of the clo ...