oc对象(包括类对象和实例对象)调用方法,主要有3个步骤,分别是消息发送,动态方法解析,消息转发
1.消息发送
2.动态方法解析
说明
其中YMPerson类的.h文件中声明一个 -run方法,但.m文件没有进行方法的实现。如果直接调用YMPerson实例对象的run方法,会报找不到方法的崩溃。但在.m实现了+resolveInstanceMethod,并在其中动态添加了方法之后,实例对象会调用添加的方法。当调用类方法的时候,如果类方法没有实现,也可以通过实现+resolveClassMethod,添加相关类方法实现,就会调用添加的类方法。代码
新建一个类
.h
#import <Foundation/Foundation.h>
NS_ASSUME_NONNULL_BEGIN
@interface YMPerson : NSObject
- (void)run;
@end
NS_ASSUME_NONNULL_END
.m
#import "YMPerson.h"
#import <objc/runtime.h>
@implementation YMPerson
+ (BOOL)resolveInstanceMethod:(SEL)sel {
if (sel == @selector(run)) {
Method method = class_getInstanceMethod(self, @selector(test));
class_addMethod(self, sel, method_getImplementation(method), method_getTypeEncoding(method));
return YES;
}
return [super resolveInstanceMethod:sel];
}
- (void)test {
NSLog(@"%s", __func__);
}
-
运行结果
3.消息转发
- 说明
1.当动态方法解析阶段没有进行任何操作的时候,就会进入消息转发阶段,此阶段会首先调用forwardingTargetForSelector,返回值不为nil,进入消息发送阶段。如果返回值为nil,会调用methodSignatureForSelector。
2.methodSignatureForSelector方法返回值为nil,调用doesNotRecognizeSelector,返回值不为nil,调用forwardInvocation方法。
1.当调用forwardingTargetForSelector,会要求返回一个target,然后会去寻找这个target中对应的方法,这个方法必须要实现,不然报doesNotRecognizeSelector。当target中实现了指定的方法,会进行此target的消息发送。
- 代码
在刚才的代码基础上新建一个YMCat类,并实现-run方法。然后在YMPerson类中指定target,YMCat中的-run方法被调用,实现消息转发。
新建YMCat类
#import <Foundation/Foundation.h>
NS_ASSUME_NONNULL_BEGIN
@interface YMCat : NSObject
- (void)run;
@end
NS_ASSUME_NONNULL_END
#import "YMCat.h"
@implementation YMCat
- (void)run {
NSLog(@"%s",__func__);
}
@end
#import "YMPerson.h"
#import <objc/runtime.h>
#import "YMCat.h"
@implementation YMPerson
- (id)forwardingTargetForSelector:(SEL)aSelector {
if (aSelector == @selector(run)) {
return [[YMCat alloc] init];
}
return [super forwardingTargetForSelector:aSelector];
}
@end
-
运行结果
2.当forwardingTargetForSelector返回值为nil,即没有指定target时,会进入消息转发的最后阶段,调用methodSignatureForSelector,此方法会要求返回一个方法签名,此方法签名包含了方法的返回值类型、参数类型,然后调用forwardInvocation,并携带一个NSInvocation对象,对此对象指定任务target,实现消息转发
#import "YMPerson.h"
#import <objc/runtime.h>
#import "YMCat.h"
@implementation YMPerson
- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector {
if (aSelector == @selector(run)) {
// v @ : 分别b表示返回值为void,还包含-run的两个默认参数,id类型self, SEL类型_cmd
return [NSMethodSignature signatureWithObjCTypes:"v@:"];
}
return [super methodSignatureForSelector:aSelector];
}
- (void)forwardInvocation:(NSInvocation *)anInvocation {
/// 指定任务target 调用YMCat中的-run 从而实现消息转发
[anInvocation invokeWithTarget:[[YMCat alloc] init]];
}
@end
-
运行结果
参考:MJ老师课程。