小码哥底层原理笔记:Runtime之Method

我们接下来看看类对象的本质,其实就是下面这个结构体:

struct objc_class : objc_object {
    Class isa;//这个isa指针本来是在objc_object里面的,现在把它拿上来这里
    Class superclass;//指向父类的指针
    cache_t cache;             // 方法缓存
    class_data_bits_t bits;    // class_rw_t * plus custom rr/alloc flags
}

最后一个bits存储着非常多的东西,跟之前说的位域一样,想要取出某些东西必须bits&XX_MASK掩码。比如我们要取出这个类对象里面存储的data数据class_rw_t,则必须bits&FAST_DATA_MASK得到class_rw_t。

struct class_rw_t {//re=read write可读写的
    // Be warned that Symbolication knows the layout of this structure.
    uint32_t flags;
    uint32_t version;

    const class_ro_t *ro;//ro=read only是只读的,无法修改的

    method_array_t methods;//方法列表,如果是元类对象,那么放的就是类方法
    property_array_t properties;//属性列表
    protocol_array_t protocols;//协议列表

    Class firstSubclass;
    Class nextSiblingClass;

    char *demangledName;
}

这个class_rw_t是可读写的,我们runtime运行时可以动态读写里面的信息,比如增加协议,方法,属性等。
class_ro_t的结构如下:

struct class_ro_t {
    uint32_t flags;
    uint32_t instanceStart;
    uint32_t instanceSize;//instance对象占用的内存空间
#ifdef __LP64__
    uint32_t reserved;
#endif

    const uint8_t * ivarLayout;
    
    const char * name;//类名
    method_list_t * baseMethodList;//方法列表
    protocol_list_t * baseProtocols;//协议列表
    const ivar_list_t * ivars;//成员变量列表

    const uint8_t * weakIvarLayout;
    property_list_t *baseProperties;

    method_list_t *baseMethods() const {
        return baseMethodList;
    }
};
OC类对象的底层结构

class_rw_t里面的methods、properties、protocols是二维数组,是可读可写的,包含了类的初始内容、分类的内容。

比如methods里面存放的是method_list_t数组,method_list_t数组里面装的才是method_t

class_ro_t里面存放的baseMethodList、baseProtocols、ivars是类里面本身初始化的方法、协议和属性不包括分类的,这些是一维数组,是不可更改的。

当我们的程序运行起来的时候系统会将class_ro_t里面的方法生成一个method_list_t数组放到class_rw_t的methods里面,并且分类里面的方法数组也会生成一个method_list_t合并到class_rw_t的methods里面,我们运行时动态修改方法就是只能修改class_rw_t里面的方法。

iOS方法的内部实现实际上是这样:

struct method_t {
    SEL name;//函数名
    const char *types;//编码(返回值类型、参数类型)
    IMP imp;//指向函数的指针(函数地址)
}

SEL name函数名其实就是@selector(test),说白了这个就是函数名字符串
const char *types编码包含了函数的返回值类型和参数类型,比如以下test方法:

- (void) test{}

它的types是v16@0:8,v表示返回值类型是void,@表示id类型,:代表SEL类型,每个OC方法底层默认都会自带两个参数-(void)test:(id)self _cmd:(SEL)_cmd{}

方法缓存 cache_t cache

cache_t cache; 是用一个散列表将曾经调用过的方法缓存起来,下次调用的时候直接从缓存里面拿,从基类里面找到的方法也会缓存到它的类对象里面,这样下次调用就不用一层一层去找了。

cache_t结构如下:

struct cache_t {
    struct bucket_t *_buckets;//散列表
    mask_t _mask;//散列表的长度 -1
    mask_t _occupied;// 以及缓存的方法数量
}

struct bucket_t {
    cache_key_t _key;//SEL作为Key即@selector(test)
    IMP _imp;//函数的内存地址
}

当我们在散列表找方法时,我们拿@selector(test) & _mask得到对应的索引,然后拿函数地址与对应索引的函数地址比较,如果不对,那么_mask减1,如果_mask为0,则_mask就赋值为散列表的长度继续找。如果散列表扩容了,那么编译器会清空缓存,即清空散列表,然后再重新存取。

OC方法执行流程:

OC调用一个方法实质就是通过objc_msgSend方法向这个对象发送一条消息。主要有三大流程:消息查找发送、动态方法解析、消息转发
消息发送

消息发送

动态方法解析
动态方法解析

消息转发
消息转发

方法替换

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 211,884评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,347评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,435评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,509评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,611评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,837评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,987评论 3 408
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,730评论 0 267
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,194评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,525评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,664评论 1 340
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,334评论 4 330
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,944评论 3 313
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,764评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,997评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,389评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,554评论 2 349

推荐阅读更多精彩内容