iOS高性能OC三:Runtime Message

2020-08-17  本文已影响0人  Trigger_o

1.消息发送objc_msgSend
OC中在运行期决定调用什么方法,方法的调用转换成C函数

//#import <objc/message.h>
    objc_msgSend(obj, @selector(messageName:), parameter);

这个函数的参数个数可变,第一个是接收消息的对象,第二个是方法名,后面的是参数,顺序和转换前的OC方法一样.

2.消息转发
前面说到消息的传递,如果objc_msgSend最终都没有找到对应的函数,也就是对象收到了无法解读的消息,这时候就会启动消息转发机制.
分为两大阶段,第一阶段,先看类能不能动态添加方法来处理这条消息,叫做动态方法解析;第二阶段,第一步先找有没有其他对象可以处理,如果有则转发给那个对象,如果没有则启动完整消息转发,消息会被封装到NSInvocation对象中,再一次询问接受者能否处理.

void newMethod(id self, SEL _cmd, id value){
    NSLog(@"this is new method -- %@",value);
}

+ (BOOL)resolveInstanceMethod:(SEL)sel{
    NSLog(@"undefind method");
    class_addMethod(self, sel, (IMP)newMethod, "v@:@");
    return YES;
}

Example *exam = [[Example alloc]init];
    NSString *exstr = (NSString *)exam;
    [exstr isEqualToString:@"abc"];

这是一个继承自NSObject的类,将其对象转换成NSString,然后调用isEqualToString方法,显然这个类是没有这个方法的,于是就会走resolveInstanceMethod方法,还能看到"abc"打印出来,如果调用的方法没有参数,打印出来的就是null


打印出参数

并且resolveInstanceMethod可以用于实现@dynamic的setter和getter,CALayer可以添加属性也是这么添加属性的

- (id)forwardingTargetForSelector:(SEL)aSelector{
    return @"abc";
}

Example *exam = [[Example alloc]init];
    NSString *exstr = (NSString *)exam;
    if([exstr isEqualToString:@"abc"]){
        NSLog(@"forward target");
    };
打印结果
//首先要实现这个方法
- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector{
    if(![self respondsToSelector:aSelector]){
        //创建一个NSMethodSignature 绑定需要转换的方法
        return [[self class] instanceMethodSignatureForSelector:@selector(newMethod:)];
    }
   return nil;
}

//如果实现了methodSignatureForSelector 会走这个方法
- (void)forwardInvocation:(NSInvocation *)anInvocation{
    //NSMethodSignature的方法要和anInvocation一致,这一步还可以修改参数
    anInvocation.selector = @selector(newMethod:);
    [anInvocation invoke]; //  执行更换后的方法
}

- (void)newMethod:(NSString *)str{
    NSLog(@"newMethod string = %@",str);
}

如果想要通过更换选择器来处理未知消息,[anInvocation invoke]也可以不执行,或者forwardInvocation里什么都不写,相当于忽略了这条消息.但是无论forwardInvocation里写不写代码,methodSignatureForSelector中一定要换掉消息的选择器.

NSInvocation的详细使用方法如下:

//NSInvocation;用来包装方法和对应的对象,它可以存储方法的名称,对应的对象,对应的参数,
    /*
     NSMethodSignature:签名:再创建NSMethodSignature的时候,必须传递一个签名对象,签名对象的作用:用于获取参数的个数和方法的返回值
     */
    //创建签名对象的时候不是使用NSMethodSignature这个类创建,而是方法属于谁就用谁来创建,是NSObject的方法
    NSMethodSignature*signature = [[self class] instanceMethodSignatureForSelector:@selector(sendMessageWithNumber:WithContent:)];
    //1、创建NSInvocation对象
    NSInvocation*invocation = [NSInvocation invocationWithMethodSignature:signature];
    invocation.target = self;
    //invocation中的方法必须和签名中的方法一致。
    invocation.selector = @selector(sendMessageWithNumber:WithContent:);
    /*第一个参数:需要给指定方法传递的值
           第一个参数需要接收一个指针,也就是传递值的时候需要传递地址*/
    //第二个参数:需要给指定方法的第几个参数传值
    NSString*number = @"1111";
    //注意:设置参数的索引时不能从0开始,因为0已经被self占用,1已经被_cmd占用
    [invocation setArgument:&number atIndex:2];
    NSString*number2 = @"啊啊啊";
    [invocation setArgument:&number2 atIndex:3];
    //2、调用NSInvocation对象的invoke方法
    //只要调用invocation的invoke方法,就代表需要执行NSInvocation对象中指定对象的指定方法,并且传递指定的参数
    [invocation invoke];
消息转发流程

在这个流程中,resolveInstanceMethod返回YES也不一定会结束,关键看这个方法中,是否添加了函数去处理这个消息,如果什么都不写,直接reture YES也一样会执行后续的流程.

3.交换调配method swizzling

//函数指针
id (*IMP)(id,SEL,...)

/*互换方法实现*/
//获取方法实现
    Method method1 = class_getInstanceMethod(NSString.class, @selector(ex_uppercaseString));
    Method method2 = class_getInstanceMethod(NSString.class, @selector(uppercaseString));
    //交换方法实现
    method_exchangeImplementations(method1, method2);

//扩展方法,写在分类中
- (NSString *)ex_uppercaseString{
    NSString *str = [self ex_uppercaseString];
    /*
     ...
     */
    NSLog(@"扩展方法--%@",str);
    return str;
}

这个方法可以将那些看不到源码的黑盒方法进行一些扩展,例如增加日志,有助于调试.

上一篇下一篇

猜你喜欢

热点阅读