iOS用runtime给一个类动态添加方法class_addMethod

介绍下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 
@interface Person : NSObject
@end

在Person.m中用class_addMethod动态添加方法

#import "Person.h"
#import 
@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”很好的文章

消息转发

截屏2019-07-07下午10.51.14.png

(疑问??疑问??)如果在这一步也不处理,只要你实现forwardInvocation :方法就不会抛出异常,消息被过滤掉,也就是并不会走doesNotRecognizeSelector:方法。?????这里和下图forwardInvocation 不处理时是否会抛doesNotRecognizeSelector

截屏2019-07-07下午10.53.17.png

我们都知道调用一个没有实现的方法时,会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:方法。

你可能感兴趣的:(iOS用runtime给一个类动态添加方法class_addMethod)