iOS开发-消息转发
消息转发是OC运行时比较重要的特性,Objective-C运行时的主要的任务是负责消息分发,我们在开发中"unrecognized selector sent to instance xx",实例对象没有实现对应的消息,通常我们只需要实现未实现的方法即可。一般情况我们处理一个方法,运行时寻找匹配的selector然后执行,但是有时候只想在运行时才创建某个方法,消息确没有具体的实现,这个时候就会出出现运行时错误,按照消息转发的顺序我们有三种解决办法。
动态方法处理
首先我们来看一个简单的例子,定义一个Message类,定义一个responseMethod方法,不实现方法,直接调用:
Message *msg=[[Message alloc]init];
[msg responseMethod];
错误信息如下:
*** Terminating app due to uncaught exception 'NSInvalidArgumentException', reason: '-[Message responseMethod]: unrecognized selector sent to instance 0x10011ad20'
动态方法的重写有两个可以调用方法:
+ (BOOL)resolveClassMethod:(SEL)sel __OSX_AVAILABLE_STARTING(__MAC_10_5, __IPHONE_2_0);
+ (BOOL)resolveInstanceMethod:(SEL)sel __OSX_AVAILABLE_STARTING(__MAC_10_5, __IPHONE_2_0);
此时我们只需要需要重写+resolveInstanceMethod:
返回YES就可以解决错误信息,注意我们需要通过class_method方法添加新的函数取代原有的sel:
+(BOOL)resolveInstanceMethod:(SEL)sel{
NSLog(@"FlyElephant-http://www.cnblogs.com/xiaofeixiang/");
if (sel==@selector(responseMethod)) {
class_addMethod([self class],sel, (IMP) dynamicMethodIMP, "v@:");
return YES;
}
return [super resolveClassMethod:sel]; }
+(BOOL)resolveClassMethod:(SEL)sel{
return [super resolveClassMethod:sel];
}
动态执行的函数:
void dynamicMethodIMP(id self, SEL _cmd)
{
NSLog(@"Developer--dynamicMethodIMP--%@",NSStringFromSelector(_cmd));
}
消息转发
如果上面的方法没有重写或者说是返回NO,那么我们接下来的按照顺序还有两种选择,两种选择的原理都是一样,消息有对应的target,我们需要更换对应的target即可:
- (id)forwardingTargetForSelector:(SEL)aSelector __OSX_AVAILABLE_STARTING(__MAC_10_5, __IPHONE_2_0);
forwardingTargetForSelector返回参数是一个对象,如果对象非nil、非self,系统会将运行的消息转发给这个对象执行,否则会执行第三种解决方案。
-(id)forwardingTargetForSelector:(SEL)aSelector{
NSLog(@"FlyElephant-http://www.cnblogs.com/xiaofeixiang/");
NSLog(@"forwardingTargetForSelector");
if (aSelector==@selector(responseMethod)) {
return developer;
}
return self;
}
第二种消息是将消息发送到另外一个对象,如果想要修改消息,那么就要使用-forwardInvocation:
,运行时将消息打包成NSInvocation,然后返回给你处理。处理完之后,调用i
nvokeWithTarget,但是如果是是调用-forwardInvocation是无法执行成功,在执行之前我们进行方法签名。
-(NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector{
NSLog(@"methodSignatureForSelector");
if ([super respondsToSelector:aSelector]) {
return [super methodSignatureForSelector:aSelector];
}else{
return [developer methodSignatureForSelector:aSelector];
}
} -(void)forwardInvocation:(NSInvocation *)anInvocation{
NSLog(@"forwardInvocation");
SEL sel=[anInvocation selector];
if ([developer respondsToSelector:sel]) {
[anInvocation invokeWithTarget:developer];
}else{
[super forwardInvocation:anInvocation];
}
}
关于Developer类中的方法:
@implementation Developer -(void)responseMethod{
NSLog(@"博客园:FlyElephant-http://www.cnblogs.com/xiaofeixiang/");
NSLog(@"Developer----responseMethod");
} @end
Cocoa中代理(Proxies)和响应链(Responder Chain)用到了消息转发。NSProxy是一个轻量级的class,它的作用就是转发消息到另一个object。如果想要惰性加载object的某个属性会很有用。NSUndoManager也有用到,不过是截取消息,之后再执行,而不是转发到其他的地方。
响应链是关于Cocoa如何处理与发送事件与行为到对应的对象,通常我们处理的键盘文本框事件First Responder,如果没有处理该消息,则转发到下一个-nextResponder
。这么一直下去直到找到能够处理该消息的object,或者没有找到,报错。
iOS开发-消息转发的更多相关文章
- iOS的消息转发机制详解
iOS开发过程中,有一类的错误会经常遇到,就是找不到所调用的方法,当然这类问题比较好解决,给当前对象或其父类对象添加该方法即可,使得编译器在编译时能正确找到该方法:或者,还有另外的方法,由于Objec ...
- iOS开发——消息推送跳转
项目开发用集成是极光推送JPush 这里主要是消息推送过来处理对应界面跳转 同时看到两篇写的不错的相关博客分享一下: http://www.jianshu.com/ ...
- iOS开发-消息通知机制(NSNotification和NSNotificationCenter)
iOS中委托模式和消息机制基本上开发中用到的比较多,一般最开始页面传值通过委托实现的比较多,类之间的传值用到的比较多,不过委托相对来说只能是一对一,比如说页面A跳转到页面B,页面的B的值改变要映射到页 ...
- iOS开发消息推送原理
转载自:http://www.cnblogs.com/cdts_change/p/3240893.html 一.消息推送原理: 在实现消息推送之前先提及几个于推送相关概念,如下图1-1: 1.Prov ...
- 玩转iOS开发 - 消息推送
消息推送
- iOS开发-消息初认识
一.消息循环(runLoop)的作用 1,防止程序退出, 2,接受事件 3,如果没有事件,让程序自动休眠 二.消息源 1, 输入源:键盘.鼠标.NSBoard.NSPort 2,定时源 ...
- iOS 消息转发机制
这篇博客的前置知识点是 OC 的消息传递机制,如果你对此还不了解,请先学习之,再来看这篇.这篇博客我尝试用口语的方式像讲述 PPT 一样给大家讲述这个知识点. 我们来思考一个问题,如果对象在收到无法解 ...
- iOS开发·runtime原理与实践: 消息转发篇(Message Forwarding) (消息机制,方法未实现+API不兼容奔溃,模拟多继承)...
本文Demo传送门: MessageForwardingDemo 摘要:编程,只了解原理不行,必须实战才能知道应用场景.本系列尝试阐述runtime相关理论的同时介绍一些实战场景,而本文则是本系列的消 ...
- iOS开发之线程间的MachPort通信与子线程中的Notification转发
如题,今天的博客我们就来记录一下iOS开发中使用MachPort来实现线程间的通信,然后使用该知识点来转发子线程中所发出的Notification.简单的说,MachPort的工作方式其实是将NSMa ...
随机推荐
- passive 的事件监听器(转载)
passive 的事件监听器 很久以前,addEventListener() 的参数约定是这样的: addEventListener(type, listener, useCapture) 后来,最后 ...
- VeeValidate配置中文的两种方法
使用VeeValidate时遇到的问题,下面是我找到的一些解决办法: VeeValidate一直报错早不到addlocale方法 解决办法:1.卸载掉当前版本,重新安装低版本如2.0.0-rc.25 ...
- 并发之atomicInteger与CAS机制
并发之atomic与CAS自旋锁 通过前几章的讲解我们知道i++这种类似操作是不安全的.针对这种情况,我们可能会想到利用synchronize关键字实现线程同步,保证++操作的原子性,的确这是一种有效 ...
- Codeforces 5C Longest Regular Bracket Sequence(DP+括号匹配)
题目链接:http://codeforces.com/problemset/problem/5/C 题目大意:给出一串字符串只有'('和')',求出符合括号匹配规则的最大字串长度及该长度的字串出现的次 ...
- (七)CXF添加拦截器
今天开始讲下拦截器,前面大家学过servlet,struts2 都有拦截器概念,主要作用是做一些权限过滤,编码处理等: webservice也可以加上拦截器,我们可以给webservice请求加权限判 ...
- jmeter在返回的json串中提取需要的值
接口测试时我们需要对某条信息进行修改,如我们先创建了一篇文章,然后进行修改操作 我们就需要找到该文章的唯一标志,如id 示例:我们要将下图返回的json 中id进行提取 注:可输入$.加需要的key即 ...
- Number对象的常用属性和方法
方法 描述 isNan() 检查值是否为数字 toFied() 将特定数字四舍五入至小数位数(返回一个字符串) toPrecision() 按数字的位数四舍五入(返回一个字符串) toExponent ...
- 微信小程序~wx.getUserInfo逐渐废弃,小程序登录过程将如何优化?
很多的时候我们在做小程序应用的时候,希望用户在使用小程序前进行登录授权,之前登录后通过wx.getUserInfo直接弹出授权的登录方式官方的意思是将不再支持,而是让用户通过下面的方式授权用户信息 & ...
- jquery attr方法和prop方法获取input的checked属性问题
jquery attr方法和prop方法获取input的checked属性问题 问题:经常使用jQuery插件的attr方法获取checked属性值,获取的值的大小为未定义,此时可以用prop方法 ...
- Python高级正则
import re p = re.compile("^[0-9]") m = p.match('13435aSAdb') print(m.group()) 一.上面的第二行和第三行 ...