iOS 底层 - runtime之objc_msgSend

2020-04-02  本文已影响0人  水中的蓝天

本文源自本人的学习记录整理与理解,其中参考阅读了部分优秀的博客和书籍,尽量以通俗简单的语句转述。引用到的地方如有遗漏或未能一一列举原文出处还望见谅与指出,另文章内容如有不妥之处还望指教,万分感谢 !

Runtime的消息机制

方法调用

在OC中的方法调用都转换为运行时API--> objc_msgSend()函数的调用; objc_msgSend如果找不到合适的方法进行调用,会报unrecognized selector sent to instance(该方法找不到)的错误

关于SEL

objc_msgSend()的执行流程三阶段:消息发送动态方法解析消息转发

消息发送

消息发送流程@2x.png

逻辑解读:

动态方法解析

动态方法解析@2x.png

底层源码:

/***********************************************************************
* _class_resolveMethod
* Call +resolveClassMethod or +resolveInstanceMethod.
* Returns nothing; any result would be potentially out-of-date already.
* Does not check if the method already exists.
**********************************************************************/
static void
_class_resolveMethod(id inst, SEL sel, Class cls)
{
    if (! cls->isMetaClass()) {
        // try [cls resolveInstanceMethod:sel]
        _class_resolveInstanceMethod(inst, sel, cls);
    } 
    else {
        // try [nonMetaClass resolveClassMethod:sel]
        // and [cls resolveInstanceMethod:sel]
        _class_resolveClassMethod(inst, sel, cls);
        if (!lookUpImpOrNil(inst, sel, cls)) {
            _class_resolveInstanceMethod(inst, sel, cls);
        }
    }
}

逻辑解读:

#import <objc/runtime.h>

###一个类只做了 - (void) test 方法的声明,没有实现

- (void)other
{
    NSLog(@"%s", __func__);
}

+ (BOOL)resolveInstanceMethod:(SEL)sel
{
    if (sel == @selector(test)) {
        // 获取其他方法
        Method method = class_getInstanceMethod(self, @selector(other));

        // 动态添加test方法的实现
        class_addMethod(self, sel,
                        method_getImplementation(method),
                        method_getTypeEncoding(method));

        // 返回YES代表有动态添加方法,注意这里返回的布尔值底层并没有做任何逻辑处理,只是哪来做一个日志输出;
        return YES;
    }
    return [super resolveInstanceMethod:sel];
}

###一个类只做了 + (void) test 方法的声明,没有实现
void c_other(id self, SEL _cmd)
{
    NSLog(@"c_other - %@ - %@", self, NSStringFromSelector(_cmd));
}

+ (BOOL)resolveClassMethod:(SEL)sel
{
    if (sel == @selector(test)) {
        // 第一个参数是object_getClass(self)
        class_addMethod(object_getClass(self), sel, (IMP)c_other, "v16@0:8");
        return YES;
    }
    return [super resolveClassMethod:sel];
}

动态添加方法

动态添加方法@2x.png

@dynamic 关键字

// 提醒编译器不要自动生成setter和getter的实现、不要自动生成成员变量
@dynamic age;
void setAge(id self, SEL _cmd, int age)
{
    NSLog(@"age is %d", age);
}

int age(id self, SEL _cmd)
{
    return 120;
}

+ (BOOL)resolveInstanceMethod:(SEL)sel
{
    if (sel == @selector(setAge:)) {
        class_addMethod(self, sel, (IMP)setAge, "v@:i");
        return YES;
    } else if (sel == @selector(age)) {
        class_addMethod(self, sel, (IMP)age, "i@:");
        return YES;
    }
    return [super resolveInstanceMethod:sel];
}

消息转发

消息转发@2x.png

知识点:

STATIC_ENTRY __objc_msgForward_impcache
    // Method cache version
    // THIS IS NOT A CALLABLE C FUNCTION
    // Out-of-band condition register is NE for stret, EQ otherwise.
    
    je  __objc_msgForward_stret

       // int __forwarding__
    jmp __objc_msgForward    

    END_ENTRY __objc_msgForward_impcache

逻辑解读:

前提:在当前消息接收者体系中找不到方法的具体实现(自己没有能力来处理,将消息转发给别人来处理)

doesNotRecognizeSelector:源码实现

// Replaced by CF (throws an NSException)
- (void)doesNotRecognizeSelector:(SEL)sel {
    _objc_fatal(
                "-[%s %s]: unrecognized selector sent to instance %p", 
                 object_getClassName(self), 方法调用者类名
                 sel_getName(sel),  方法名
                 self                           方法调用者
                  );
}

代码示例

- (id)forwardingTargetForSelector:(SEL)aSelector
{
    if (aSelector == @selector(test)) {
        // objc_msgSend([[XYHCat alloc] init], aSelector)
        如果返回空,就会触发methodSignatureForSelector:方法
        return [[XYHCat alloc] init];
    }
    return [super forwardingTargetForSelector:aSelector];
}

 方法签名:返回值类型、参数类型
- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector
{
    if (aSelector == @selector(test)) {
        return [NSMethodSignature signatureWithObjCTypes:"v16@0:8"];
//        return [NSMethodSignature signatureWithObjCTypes:"i@:i"];
//        return [[[MJCat alloc] init] methodSignatureForSelector:aSelector];
    }
    return [super methodSignatureForSelector:aSelector];
}

 NSInvocation封装了一个方法调用,包括:方法调用者、方法名、方法参数
    anInvocation.target 方法调用者
    anInvocation.selector 方法名
    [anInvocation getArgument:NULL atIndex:0]
//此方法中的所有实现就相当于是当初调用的方法的一个实现,比如:[XYHPerson test] , test方法没有具体实现,当执行到forwardInvocation:方法时,其内部的实现最终就是test的实现
- (void)forwardInvocation:(NSInvocation *)anInvocation
{

不带参数指定target,方法调用
//    anInvocation.target = [[XYHCat alloc] init];
//    [anInvocation invoke];
    [anInvocation invokeWithTarget:[[XYHCat alloc] init]];

带参数指定target,方法调用
// 参数顺序:0  --> receiver、1 --> selector、2 --> other arguments
//    int age;
//    [anInvocation getArgument:&age atIndex:2];
//    NSLog(@"%d", age + 10);
    
    // anInvocation.target == [[XYHCat alloc] init]
    // anInvocation.selector == test:
    // anInvocation的参数:15
    // [[[XYHCat alloc] init] test:15]

    [anInvocation invokeWithTarget:[[XYHCat alloc] init]];
    
    int ret;
   //拿到返回值
    [anInvocation getReturnValue:&ret];
    
    NSLog(@"%d", ret);

}

类方法的消息转发

和对象方法一样,但在forwardingTargetForSelector:、methodSignatureForSelector:、forwardInvocation是需要执行 '+'开头的方法

+ (id)forwardingTargetForSelector:(SEL)aSelector
{
    // objc_msgSend([XYHCat class], @selector(test))
    // [[[MJCat alloc] init] test]
    if (aSelector == @selector(test)) return [XYHCat class];

    return [super forwardingTargetForSelector:aSelector];
}

疑问:如果在类方法的forwardingTargetForSelector:中返回的不是类对象,而是一个实例对象且有对应的实例方法,能够调用成功吗 ?为什么?

可以的,因为在这里返回的对象不管是什么对象,只要其已经实现了对应的方法都会通过objc_msgSend()函数调用成功,只要有这个方法就行;

@implementation XHYCat

+ (void)test
{
    NSLog(@"%s", __func__);
}

- (void)test
{
    NSLog(@"%s", __func__);
}

@end

+ (id)forwardingTargetForSelector:(SEL)aSelector
{
    if (aSelector == @selector(test)) 
    //这么写最终会执行   - (void)test
    return [[XHYCat alloc] init];
    return [super forwardingTargetForSelector:aSelector];
}

特殊的NSProxy

特殊:

  1. (NSMethodSignature *)methodSignatureForSelector:(SEL)sel方法必须实现
  2. (void)forwardInvocation:(NSInvocation *)invocation必须实现
@interface NSProxy <NSObject> {
    Class   isa;
}
@interface NSObject <NSObject> {
    Class isa  OBJC_ISA_AVAILABILITY;
}

应用:

+ (instancetype)proxyWithTarget:(id)target
{
    XYHProxy *proxy = [XYHProxy alloc];
    proxy.target = target;
    return proxy;
}

//消息转发到target

//返回方法签名
- (NSMethodSignature *)methodSignatureForSelector:(SEL)sel
{
    return [self.target methodSignatureForSelector:sel];
}

//内部实现方法调用
- (void)forwardInvocation:(NSInvocation *)invocation
{
    [invocation invokeWithTarget:self.target];
}

objc_msgSend()的执行流程 - 源码跟读

QQ20200402-161206@2x.png

ENTRY _objc_msgSend 汇编源码

    ENTRY _objc_msgSend
    UNWIND _objc_msgSend, NoFrame
//x0寄存器:消息接收者-->receiver
    cmp p0, #0          // nil check and tagged pointer check
#if SUPPORT_TAGGED_POINTERS
    b.le    LNilOrTagged        //  (MSB tagged pointer looks negative)
#else
    b.eq    LReturnZero
#endif
    ldr p13, [x0]       // p13 = isa
    GetClassFromIsa_p16 p13     // p16 = class
LGetIsaDone:
    // calls imp or objc_msgSend_uncached
    CacheLookup NORMAL, _objc_msgSend

#if SUPPORT_TAGGED_POINTERS
LNilOrTagged:
    b.eq    LReturnZero     // nil check

    // tagged
    adrp    x10, _objc_debug_taggedpointer_classes@PAGE
    add x10, x10, _objc_debug_taggedpointer_classes@PAGEOFF
    ubfx    x11, x0, #60, #4
    ldr x16, [x10, x11, LSL #3]
    adrp    x10, _OBJC_CLASS_$___NSUnrecognizedTaggedPointer@PAGE
    add x10, x10, _OBJC_CLASS_$___NSUnrecognizedTaggedPointer@PAGEOFF
    cmp x10, x16
    b.ne    LGetIsaDone

    // ext tagged
    adrp    x10, _objc_debug_taggedpointer_ext_classes@PAGE
    add x10, x10, _objc_debug_taggedpointer_ext_classes@PAGEOFF
    ubfx    x11, x0, #52, #8
    ldr x16, [x10, x11, LSL #3]
    b   LGetIsaDone
// SUPPORT_TAGGED_POINTERS
#endif

LReturnZero:
    // x0 is already zero
    mov x1, #0
    movi    d0, #0
    movi    d1, #0
    movi    d2, #0
    movi    d3, #0
    ret //ret相当于c语言的return

    END_ENTRY _objc_msgSend
 
上一篇 下一篇

猜你喜欢

热点阅读