OC class class_rw_t 原理解析

前言:

OC属于一门动态性的语言,一般语言编译流程是:编写代码->编译链接->运行代码。而OC因为有runtime机制,在运行代码的时候对编译链接动态改变。

isa指针

arm64之后,我们是无法直接获取到一个对象的isa地址,是因为class中的isa使用了一个union来储存isa还有其他的一些信息,优化内存的使用率。通过源码 我们可以看到isa的一些信息。


窥探class结构
/* isa指针结构  */
union isa_t 
{
    isa_t() { }
    isa_t(uintptr_t value) : bits(value) { }
    Class cls;
    uintptr_t bits;
#   define ISA_MASK        0x0000000ffffffff8ULL
#   define ISA_MAGIC_MASK  0x000003f000000001ULL
#   define ISA_MAGIC_VALUE 0x000001a000000001ULL
    struct {
        uintptr_t nonpointer        : 1; /* 0:代表普通指针,代表Class Meta-Class对象的内存地址; 1:代表优化过,使用位域存储更多的信息 */
        uintptr_t has_assoc         : 1;/* 是否设置过关联对象,如果没有释放会更快,有设置过则会变成1 */
        uintptr_t has_cxx_dtor      : 1;/* 是否有C++的析构函数(.cxx_destruct),如果没有释放会更快  */
        uintptr_t shiftcls          : 33; /* 储存这Class,Meta-Class的内存地址信息,因为处在第三位,所以获取到真正isa指针地址的时候,需要&一个mask值 */
        uintptr_t magic             : 6; /* 用于调试分辨对象是否完成初始化 */
        uintptr_t weakly_referenced : 1;/* 是否被弱引用过,如果没有释放更快 */
        uintptr_t deallocating      : 1;/* 对象是否正在释放 */
        uintptr_t has_sidetable_rc  : 1; /*里面储存的值使引用计数-1 */
        uintptr_t extra_rc          : 19;/* 引用技术区是否大于无法储存在isa中,如果为1,那么引用计数会储存在一个叫SideTable的类的属性中 */
    };
}

窥探class_rw_t 内部结构

class_rw_t内部结构.png

当类运行时的时候,runtime机制会将储存在ro(只读)内部的class方法合并到method_array_t methods中去,便于动态添加方法。协议,属性亦是如此。

窥探class_ro_t 内部结构

class_ro_t 内部结构

这里面的方法列表等,来源于初始化class的时候方法列表,通过runtime机制,会合并到class_rw_t(可读可写)中去,
所以class_rw_t中的methods包含class_ro_t中的methods,还包括通过runtime添加到class中去的method。

method_t 内部结构

从上面我们可以看出,存放函数放的是 数组method_list_t,和method_array_t二维数组,method_array_t存放的是method_list_t,method_list_t里面存放的是method_t结构体

struct method_t {
    SEL name; //函数名
    const char *types; //字符串编码,返回值类型
    IMP imp; //指向函数的指针

    struct SortBySELAddress :
        public std::binary_function<const method_t&,
                                    const method_t&, bool>
    {
        bool operator() (const method_t& lhs,
                         const method_t& rhs)
        { return lhs.name < rhs.name; }
    };
};

方法缓存列表 cache_t

用散列表来缓存曾经调用过的方法,可以提高查找速度

struct cache_t {
    struct bucket_t *_buckets; //散列表
    mask_t _mask;  //散列表长度-1
    mask_t _occupied; //已经缓存的方法数量
}
struct bucket_t {
private:
    cache_key_t _key; //散列表的key (SEL作为key)
    IMP _imp; //函数实现内存地址
};

从上面我可以可以大致看出缓存列表中的结构:用一个散列表来从存储方法,函数的SEL选择器作为key,来寻找IMP的实现内存地址,进行方法的调用。 牺牲内存换时间
散列表(哈希表):底层结构就是数组。
存值> 刚开始会初始化一个mask初始值,当方法调用的是时候会将@selector(method),会将@selector(method) & mask 生成一个数值索引(index)放置散列表中的索引值得位置,如果是首次引入,会index前面的置空操作留出内存以供后面操作(哈希表中用内存换取时间的概念)。不同method&mask有可能会生成相同的index,此时会将index - 1 ,继续method&mask存值,如果有值,就继续-1,如果当index = 0,还没有空位,会将index赋值给mask,从最大的索引继续寻找空位,直至找到空位,如果还没有会将进行扩容操作。此时mask2,当前数组2,清空所有值,然后在继续前面操作,直到把值放进索引中。这就是哈希表的原理
取值> 取值的时候,依旧是会将 @selector(method) & mask生成的index,然后就去对应的散列表中取值,因为赋值的时候顺序都是排列好的,所以取值的时候就比较方便了,提高效率,这样比遍历数组效率更高,更有效。

缓存验证

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

推荐阅读更多精彩内容

  • 青春年少时,一位别人眼中玩世不恭的男孩,坐在我旁边,极尽温柔地对待我、保护我,我以为那就是爱。 转眼,当初不谙世事...
    燕汐阅读 199评论 0 0
  • 雪是冬的梦 花是蝴蝶和蜂的梦 月亮是守夜人的梦 和风牵手是水的梦 谁是你的梦 希望在梦的春天里重逢
    蕙兰漱雪阅读 174评论 0 7
  • 我是在傍晚五点钟之后骑上我的自行车回家的,那时候的天还很亮,太阳好像还在西边的山头上僵持着,但是并没有多少黄昏的苍...
    冰楼阅读 276评论 0 0
  • 【编前语】几个月前写下的一篇文章,伯母丧事办完,再次想起,把简书的第一个位置给它。我知道天堂没有病痛,伯母会快乐!...
    藕仔莉阅读 1,811评论 9 16
  • 碎片化这个概念,其实最早是在华华看到某一篇文章,说利用碎片化或者是空隙的时间去给你的朋友打电话,这样时间到了你就可...
    柠檬班软件测试阅读 1,124评论 0 1