Runtime窥探 (二)| 消息发送

作者: Dely | 来源:发表于2017-09-25 15:27 被阅读141次

前言

双眸浅看,几世繁华褪色成一纸墨色,岁月索颜,在风居住的街道,我怀抱相思,在雨中等待前世,等待今世,等待晴天,亦或是等待你。

作品集

1.例子

  • 新建Person类,继承于NSObject
  • 新建方法:
- (void)eat{
    NSLog(@"实例方法正在吃饭。。。。");
}
- (void)eatWith:(NSString *)name{
    NSLog(@"实例方法正在和%@吃饭。。。。",name);
}
- (void)drink{
    NSLog(@"实例方法正在喝水。。。。");
}
    
+ (void)eat{
    NSLog(@"类方法正在吃饭。。。。");
}

看看下面例子会执行吗,如果执行打印出什么?

- (void)objc_message{
    Person *p = [Person new];
    [p eat];
    objc_msgSend(p,@selector(eatWith:),@"Dely");
    [p performSelector:@selector(eat) withObject:nil];
    [[p class] performSelector:@selector(eat) withObject:nil];
}

结果是肯定可以执行的,打印结果如下:

实例方法正在吃饭。。。。
实例方法正在和Dely吃饭。。。。
实例方法正在吃饭。。。。
类方法正在吃饭。。。。

疑问:

  • objc_msgSend是个什么东西?
  • [[p class] performSelector:@selector(eat) withObject:nil];为什么可以这样调用?

2.objc_msgSend是什么?

objc_msgSend:消息发送、负责方法的调用。

直接使用objc_msgSend需要导入头文件#import <objc/message.h>

//方法定义:消息接受者,消息,参数
objc_msgSend(id _Nullable self, SEL _Nonnull op, ...)
  • 当我们发送消息[receiver message]时,会被编译成:
objc_msgSend(receiver,@selector(message), ...)

  • 也就是说:调用[p eat]时;其实就是调用了
//无参数
objc_msgSend(p,@selector(eat));
//有参数
objc_msgSend(p,@selector(eatWith:),@"Dely");
  • performSelector方法底层其实就是基于objc_msgSend封装的,被编译成
objc_msgSend(Person,@selector(eat));

3.消息发送机制

我们就算调用了objc_msgSend方法那到底怎么调用到函数的实现的呢?他的调用机制是怎么样的呢?也就是我们常说的OC的消息发送机制.

我们在上一篇讲解isa和class的时候无形中其实也讲解了消息发送机制,主要讲解了isa和class,不明白的可以再去复习一下isa和class到底是个什么东西?

objc_msgSend如何执行方法?

上面提到了,一个OC方法被编译成objc_msgSend,那么,Runtime如何找到方法的执行体呢?

再看下runtime中objc_class定义:

//Class
typedef struct objc_class *Class;

//objc_class
struct objc_class {
    Class _Nonnull isa  OBJC_ISA_AVAILABILITY;

#if !__OBJC2__
    Class _Nullable super_class                              OBJC2_UNAVAILABLE;
    const char * _Nonnull name                               OBJC2_UNAVAILABLE;
    long version                                             OBJC2_UNAVAILABLE;
    long info                                                OBJC2_UNAVAILABLE;
    long instance_size                                       OBJC2_UNAVAILABLE;
    struct objc_ivar_list * _Nullable ivars                  OBJC2_UNAVAILABLE;
    struct objc_method_list * _Nullable * _Nullable methodLists                    OBJC2_UNAVAILABLE;
    struct objc_cache * _Nonnull cache                       OBJC2_UNAVAILABLE;
    struct objc_protocol_list * _Nullable protocols          OBJC2_UNAVAILABLE;
#endif

} OBJC2_UNAVAILABLE;

可以看到 Class 中包含了isa指针、methodLists、cache等信息。

  • 实例可以通过isa指针找到Class
  • Class中methodLists包含了类中所有的实例方法
  • 实例的元类中methodLists中包含了类中所有的加方法
objc_message

实例方法的执行过程:

例如:[p eat];

  • 1.首先被编译成objc_msgSend(id receiver, SEL op, ...);
  • 2.根据receiver对象的isa指针获取它对应的Class;
  • 3.优先在class的cache(为了提高效率)查找message方法,找到直接执行执行6,找不到继续往下执行
  • 4.再到methodLists查找,找到直接执行6,找不到继续往下执行
  • 5.没有在class找到,再到super_class查找,依次查找到基类NSObject.如果还没找到执行消息转发机制(本文不讲解)
  • 6.找到method,根据IMP执行函数,并将返回值给调用者

类方法的执行过程:

例如:[Person eat];

因为类方法列表保存在元类中,所以跟实例方法不同的是:首先类对象会根据Class的isa指针找到对应的元类。查找过程在元类中的methodLists查找方法,找不到的话,查找到当前的父类,然后在父类对应元类的methodLists。执行过程就和上面的实例方法一样了。

  • 每一个Class的isa指向唯一一个元类,每个元类的isa都指向同一个根元类
  • 其实类Class本身就是一个实例,只不过他是元类的实例。所以我们根据实例的isa就可以找到对应的类

开始还有一个疑问? [[p class] performSelector:@selector(eat) withObject:nil];为什么可以这样调用?
编译完成后都会执行objc_msgSend,只不过对象是Class类型罢了,给对象发一个消息。依次查找元类中的方法,当然可以执行了。

现在对于OC的消息转发机制是否有一些了解。上面说到消息到最后都没有找到该怎么办?他该何去何从?会立即崩溃吗?这就是下一篇博客要讲的内容:消息转发

相关文章

  • Runtime窥探 (二)| 消息发送

    前言 双眸浅看,几世繁华褪色成一纸墨色,岁月索颜,在风居住的街道,我怀抱相思,在雨中等待前世,等待今世,等待晴天,...

  • iOS - Runtime相关

    一.什么是 runtime ? 二.runtime的头文件 三.消息发送步骤 四.常用方法 五.应用

  • Runtime --- 消息发送

    上篇内容我们主要了解了objc_msgSend方法的几个参数和objc_class的结构本篇内容我们一起了解 消息...

  • runtime消息发送

    一、runtime简介 RunTime简称运行时。OC就是运行时机制,也就是在运行时候的一些机制,其中最主要的是消...

  • Runtime 消息发送

    1、isa 详解 isa 在 arm64 架构之前就是一个普通的指针,存储着 Class、Meta-Class 对...

  • Runtime消息发送

    对象方法的调用就是向这个对象发送消息 objc_method这个结构体的内容: SEL method_name 方...

  • Runtime — 消息发送

    一、Runtime 1. Runtime介绍 Objective-C 是一门动态语言,而承载整个 OC 动态特性的...

  • Runtime-原理

    runtime初探对象与方法的本质runtime-消息发送runtime-动态方法解析runtime-消息转发 r...

  • runtime objc_msgSend使用

    前言 想要通过runtime发送消息,就必须要掌握runtime如何发送消息,是调用哪个函数?又是如何调用的?本篇...

  • runtime源码之方法实现过程

    主要内容: 一:什么是 runtime? 二:对象与方法的本质 三:runtime源码解析-消息发送 四:runt...

网友评论

本文标题:Runtime窥探 (二)| 消息发送

本文链接:https://www.haomeiwen.com/subject/dktfsxtx.html