介绍下class_addMethod
runtime动态添加方法
@param cls 要添加新方法的那个类
@param name 要添加的方法
@param imp 实现这个方法的函数 ,传的类型 1,C语言写法:(IMP)方法名 2,OC的写法:class_getMethodImplementation(self,@selector(方法名:))
@param types 要添加的方法的返回值和参数 叫 type encodings
@return YES:方法添加成功 NO:方法添加失败
OBJC_EXPORT BOOL
class_addMethod(Class _Nullable cls, SEL _Nonnull name, IMP _Nonnull imp,
const char * _Nullable types);
在ViewController调Person的eat方法, 但Person没有eat方法
在OC中找不到对相应的实现方法时 有补救机制 即 会先调用动态决议方法 该方法解决不了问题 再调用重定向方法 、 消息转发
#import "ViewController.h"
#import "Person.h"
@interface ViewController ()
@end
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
Person *person = [[Person alloc]init];
//调用Person未实现方法eat
[person performSelector:@selector(eat) withObject:@"minzhe"];
}
@end
Person.h没有eat方法
#import <Foundation/Foundation.h>
@interface Person : NSObject
@end
在Person.m中用class_addMethod动态添加方法
#import "Person.h"
#import <objc/runtime.h>
@implementation Person
//动态决议方法:
+ (BOOL)resolveInstanceMethod:(SEL)sel {
Method exchangeM = class_getInstanceMethod([self class], @selector(eatWithPersonName:));
class_addMethod([self class], sel, class_getMethodImplementation(self, @selector(eatWithPersonName:)),method_getTypeEncoding(exchangeM));
return YES;
}
- (void)eatWithPersonName:(NSString *)name {
NSLog(@"Person %@ start eat ",name);
}
@end
到此实现了为Person动态添加eat方法
@ end
|
|
|
V
另一篇关于“消息转发”很好的文章
另一篇关于“RunTime”很好的文章
消息转发
(疑问??疑问??)如果在这一步也不处理,只要你实现forwardInvocation :方法就不会抛出异常,消息被过滤掉,也就是并不会走doesNotRecognizeSelector:方法。?????这里和下图forwardInvocation 不处理时是否会抛doesNotRecognizeSelector
我们都知道调用一个没有实现的方法时,会crash,我们来微笑着,一步步的看它是如何crash的,也许你还能插一手。同时想要深入灵活的了解关于函数方法的东西,我们也需要明白消息转发的机制:
消息转发第一步:+(BOOL)resolveInstanceMethod:(SEL)sel,+(BOOL)resolveClassMethod:(SEL)sel->讨薪
当向调用一个方法,但没有实现时,消息会通过上面两个方法寻找是否能找到实现?如果没有则返回NO,进入下一步。
(id)forwardingTargetForSelector:(SEL)aSelector
第一步如果返回NO会通过- (id)forwardingTargetForSelector:(SEL)aSelector方法再次寻找,不过这次找的是一个能响应该方法的对象。(NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector->私家侦探找线索
当forwardingTargetForSelector :返回nil时,会进行这一步,生成方法签名,如果方法签名为nil直接调用doesNotRecognizeSelector:返回异常,如果正常生成方法签名,则进行最后一步。(void)forwardInvocation:(NSInvocation *)anInvocation
到这这一步,其实我们还可以通过NSInvocation来力挽狂澜(我们在前面说过这个东西,也是很神奇的存在,不过有点麻烦),如果在这一步也不处理,只要你实现forwardInvocation :方法就不会抛出异常,消息被过滤掉,也就是并不会走doesNotRecognizeSelector:方法。