总所周知:OC是一门运行时语言,即其方法的派发会在运行时才确定,而静态语言则是在编译阶段就确定要调用哪些函数。
OC的动态性由runtime组件构成,通过msg_send(对象名称,方法名称)来进行运行时方法查找和调用,如果该方法没有被找到,则进入消息转发流程,这个消息转发流程可以认为是iOS提供给我们在运行时给其添加方法声明的一个方式。
消息转发分为以下步骤
第一个步骤的意思是给我们这个找不到的对象方法提供一个新的对象方法去代替他,这个是个类方法(返回值为Bool),我们可以在类里面重写他。如果我们找不到的是类方法,则可以使用resolveClassMethod:
第二个步骤的意思是我们无法通过第一步去完成,我们需要其他对象的这个方法去帮助我们完成,函数要我们返回的是一个id,即帮我们完成消息转发的这个辅助对象
第三个步骤的意思是我们没有提供新的方法去代替,也不知道有没有其他对象去帮助我们,我们出动NSInvocaiton去帮我们找,我们使用该方法传入的NSInvocation对象调用methodSignatureForSelector: 让他去帮我们查找这个方法,直到NSObject类,如果再找不到就报错。
现在我们来写一个Demo去看看,我们想要实现一个叫AutoDictionary类,而这个类可以储存我们自定义的对象。我们把其对象声明成@dynamic,则他声明的property属性不会自动生成getter和setter方法,这时候如果外部使用getter和setter方法,我们就需要在消息转发中实现getter和setter方法
AutoDictionary.h文件
#import <Foundation/Foundation.h>
@class Person;
NS_ASSUME_NONNULL_BEGIN
@interface AutoDictionary : NSObject
@property(nonatomic,strong)NSNumber *number;
@property(nonatomic,strong)NSDate *date;
@property(nonatomic,strong)id opaqueObject;
//我们可以存个person(自定义)对象进字典
@property(nonatomic,strong)Person *people;
@end
NS_ASSUME_NONNULL_END
AutoDictionary.m 文件
#import "AutoDictionary.h"
#import <objc/runtime.h>
@interface AutoDictionary ()
//显然我们需要一个字典去完成键值的存储
@property(nonatomic,strong)NSMutableDictionary *backStore;
@end
@implementation AutoDictionary
@dynamic number,date,opaqueObject;
-(id)init{
if (self= [super init]) {
_backStore = [NSMutableDictionary new];
}
return self;
}
id autoDictionaryGetter(id self,SEL _cmd){
AutoDictionary * typedSelf = (AutoDictionary*)self;
NSMutableDictionary *backingstore = typedSelf.backStore;
//我们其实可以typedSelf.backStore[key]直接返回,这里创建多一个变量好理解点?
//get方法名字就是属性名字 比如date的getter方法名字就是date
NSString *key = NSStringFromSelector(_cmd);
return backingstore[key];
}
void autoDictionarySetter(id self,SEL _cmd,id value){
AutoDictionary *typedSelf = (AutoDictionary*)self;
NSMutableDictionary *backingstore = typedSelf.backStore;
//这里使用mutableCopy让其返回一个可修改的字符串,因为我们后面要去掉他里面一些set,:和修改大小写等东西,只留下他的属性名字
NSMutableString *key = [NSStringFromSelector(_cmd) mutableCopy];
//我们这里set方法的名字:如果是date 则为setDate: 我们要把它变成date,这样才能把他作为key从字典中查找
//把set去掉 , 这边的Range采用的是右闭包
[key deleteCharactersInRange:NSMakeRange(0, 3)];
//把:冒号去掉
[key deleteCharactersInRange:NSMakeRange(key.length-1, 1)];
//把第一个字母的大写变成小写
[key replaceCharactersInRange:NSMakeRange(0, 1) withString:[[key substringToIndex:1] lowercaseString]];
if (value){
[backingstore setObject:value forKey:key];
}else{
[backingstore removeObjectForKey:key];
}
}
//1,第一步找不到该方法的实现,我们可以在这里添加该方法的实现
+ (BOOL)resolveInstanceMethod:(SEL)sel
{
//我们这里用hasPrefix @"set"来判断,因为get的方法名字就是属性名字
if (![NSStringFromSelector(sel) hasPrefix:@"set"])
{ //参数1:所属类
//参数2:找不到的方法名称
//参数3:你为找不到的方法添加的imp(实现)
//参数4:告诉系统该函数的声明类型,这个可以认为是带参数还是不带参数的方法,有不同的表明规则。
class_addMethod(AutoDictionary.class, sel, (IMP)autoDictionaryGetter, "v@:@");
return true;
}else
{
class_addMethod(AutoDictionary.class, sel, (IMP)autoDictionarySetter, "@@:");
return true;
}
}
//2,resolveInstanceMethod找不到返回错误,及在该类没有提供该方法
//我们需要传另一个对象去帮组完成该方法的调用(返回其id)
//- (id)forwardingTargetForSelector:(SEL)aSelector
//{
//// return 帮你调用该方法的函数
//}
//3,完整的消息转发
//这时候你没有提供辅助对象去完成操作,这时候是消息转发的最后一步了,亲自使用NSInvocation去帮你寻找你需要的方法,如果一直找不到,那么就会报错,一般我们会在前两步就解决掉,最好第一步,第三步通过可以通过类似第二步的方法转换一下目标去查找,第三步还要创建一个Invocation方法
- (void)forwardInvocation:(NSInvocation *)anInvocation
{
// [anInvocatio methodSignatureForSelector:@selector(你提供的方法名字)]
}
@end
假如我们加上了@dynamic而且在消息转发流程没有处理相关的函数转发
那么下面调用会报错
实现消息转发后
打断点进入resolveInstanceMethod:可以看到会停留在那边