RunTime之NSObject解析

写在前边

上篇文章中,介绍了Objc对象的分类:实例对象、类对象、元类对象;也介绍了对象分类中通过isasuperclass进行方法查找的流程。今天通过对苹果官方源码objc4-750对其详细说明。

源码

我们知道,Objective-C中,通常情况下,我们新建类都会继承于NSObject。那么,我们就从NSObject开始吧。

NSObject.png

我们看到在NSObject中,包含一个Class isa成员变量;

Class

接上图,然后我们点击能进入查看Class的定义。发现她是一结构体objc_class指针。

Class.png

同时,我们也看到了我们常用的id其实是一个叫objc_object的结构体指针。继续,查看objc_class的实现

objc_class.png

我们看到,objc_class继承于objc_object,而且还包含了其他一些信息,稍后再详细介绍这些字段具体作用。

而从前边的截图中,我们也看到了objc_object就是id类型!也就是说 Class继承于id

那么objc_object中包含什么信息呢?

id

继续,查看objc_object信息。

objc_object.png

我们在objc_object中看到了一个isa_t类别的成员变量,终于找到了传说中的isa异常兴奋!

isa_t

那么,isa_t具体是什么类型呢,继续我们发现,她原来是一个union 共用体,不仅可以用来存储类的指针信息,还可以用来表示位域(或者位段,可以用来存储更多的信息)。

isa_t-1.png

位域详情定义如下:

isa_t-2.png

这里介绍下各个位的作用

  • nonpointer:0,代表普通的指针,存储着Class、Meta-Class对象的内存地址;1,代表优化过,使用位域存储更多的信息
  • has_assoc:是否有设置过关联对象,如果没有,释放时会更快
  • has_cxx_dtor:是否有C++的析构函数(.cxx_destruct),如果没有,释放时会更快
  • shiftcls:存储着Class、Meta-Class对象的内存地址信息
  • magic:用于在调试时分辨对象是否未完成初始化
  • weakly_referenced:是否有被弱引用指向过,如果没有,释放时会更快
  • deallocating:对象是否正在释放
  • extra_rc:里面存储的值是引用计数器
  • has_sidetable_rc:引用计数器是否过大无法存储在isa中;如果为1,那么引用计数会存储在一个叫SideTable的类的属性中

分析

isa_t的位域

首先,使用使用位域可以节约资源。例如,我们比较两个类实例占用的空间。

@interface Person1 : NSObject
@property (nonatomic, assign) BOOL tall;
@property (nonatomic, assign) BOOL handsome;
@property (nonatomic, assign) NSInteger age;
@end

@interface Person2 : NSObject{
    union{
        NSInteger bits;
        struct {
            char tall : 1;
            char handsome: 1;
            NSInteger age: 7;
        } ;
    }tallHandsomeAge;
}
-(void)setTall:(BOOL)tall;
-(BOOL)tall;
-(void)setHandsome:(BOOL)handsome;
-(BOOL)handsome;
-(void)setAge:(NSInteger)age;
-(NSInteger)age;
-(NSInteger)bits;
@end

实现代码,如下:

@implementation Person1

@end

#define Tall_Mask (1<<0)
#define handsome_Mask (1<<1)
#define age_offset 2
#define age_Mask (1<<8)

@implementation Person2

-(void)setTall:(BOOL)tall{
    if (tall) {
        tallHandsomeAge.bits |= Tall_Mask;
    }else{
        tallHandsomeAge.bits &= ~Tall_Mask;
    }
}
-(BOOL)tall{
    return !!(tallHandsomeAge.bits & Tall_Mask);
}
-(void)setHandsome:(BOOL)handsome{
    if (handsome) {
        tallHandsomeAge.bits |= handsome_Mask;
    }else{
        tallHandsomeAge.bits &= ~handsome_Mask;
    }
}
-(BOOL)handsome{
    return !!(tallHandsomeAge.bits & handsome_Mask);
}
-(void)setAge:(NSInteger)age{
    tallHandsomeAge.bits &= (age_Mask | Tall_Mask | handsome_Mask);
    tallHandsomeAge.bits |= (age << age_offset);
}
-(NSInteger)age{
    return tallHandsomeAge.bits & age_Mask;
}
-(NSInteger)bits{
    return tallHandsomeAge.bits;
}
@end

首先,占用空间大小比较优势很明显;其次,位域使用的位运算对处理器来说,效率是很高的

bit_field.png
isa_t的Class(objc_class)

首先,objc_class继承自 objc_object(objc_object包含一个isa_t的成员变量isa)。objc_class结构体中包含的主要信息如下:

    // Class ISA;
    Class superclass;
    cache_t cache;             // formerly cache pointer and vtable
    class_data_bits_t bits;    // class_rw_t * plus custom rr/alloc flags

    class_rw_t *data() { 
        return bits.data();
    }
class_rw_t.png
class_ro_t.png
  • ISA指针
  • superclass
  • cache方法缓存
  • bits具体的类信息(结构体class_rw_t)
    • 方法列表
    • 属性列表
    • 协议列表
    • class_ro_t *ro
      • instance对象占用的内存大小等
      • 成员变量列表
cache方法缓存

Class中有个方法缓存cache_t,用哈希表来缓存曾经调用过的方法,以提高方法的查找速度。

struct cache_t {
    struct bucket_t *_buckets;//散列表
    mask_t _mask;//总长度
    mask_t _occupied;//已缓存的方法数量
 
    //other information
}


struct bucket_t {
private:
    // IMP-first is better for arm64e ptrauth and no worse for arm64.
    // SEL-first is better for armv7* and i386 and x86_64.
#if __arm64__
    MethodCacheIMP _imp;//函数的内存地址
    cache_key_t _key;//SEL
#else
    cache_key_t _key;//SEL
    MethodCacheIMP _imp;//函数的内存地址
#endif
    //other information
}

之前,我们介绍过objc_msgSend消息对应方法查找流程。这里再简单介绍下:

  • 先查找自己类的缓存中是否有对应的方法,如果能找到直接发消息调用
  • 查找自己类中的bits类信息中的方法列表,如果能找到,则发消息,将方法放入方法缓存中。
  • 遍历父类的方法。
bucket_t * cache_t::find(cache_key_t k, id receiver)
{
    assert(k != 0);

    bucket_t *b = buckets();
    mask_t m = mask();
    mask_t begin = cache_hash(k, m);
    mask_t i = begin;
    do {
        if (b[i].key() == 0  ||  b[i].key() == k) {
            return &b[i];
        }
    } while ((i = cache_next(i, m)) != begin);

    // hack
    Class cls = (Class)((uintptr_t)this - offsetof(objc_class, cache));
    cache_t::bad_cache(receiver, (SEL)k, cls);
}

#if __arm__  ||  __x86_64__  ||  __i386__
// objc_msgSend has few registers available.
// Cache scan increments and wraps at special end-marking bucket.
#define CACHE_END_MARKER 1
static inline mask_t cache_next(mask_t i, mask_t mask) {
    return (i+1) & mask;
}

#elif __arm64__
// objc_msgSend has lots of registers available.
// Cache scan decrements. No end marker needed.
#define CACHE_END_MARKER 0
static inline mask_t cache_next(mask_t i, mask_t mask) {
    return i ? i-1 : mask;
}

#else
#error unknown architecture
#endif
bits具体的类信息class_rw_t

这里主要介绍的是方法列表 method_array_t methods 。具体method_t的结构如下:

struct method_t {
    SEL name;//函数名称
    const char *types;//函数编码,在swazzing中用的比较多
    MethodListIMP 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; }
    };
};

参照

https://opensource.apple.com/tarballs/objc4/

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

推荐阅读更多精彩内容