亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

iOS內(nèi)存管理引用計(jì)數(shù)示例分析

 更新時(shí)間:2023年01月06日 08:55:09   作者:山海飛鳥(niǎo)  
這篇文章主要為大家介紹了iOS內(nèi)存管理引用計(jì)數(shù)示例分析詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪

內(nèi)存管理機(jī)制

目前流行的內(nèi)存管理機(jī)制主要有GCRC兩種。

  • GC (Garbage Collection):垃圾回收機(jī)制,定期查找不再使用的對(duì)象,釋放對(duì)象占用的內(nèi)存。
  • RC (Reference Counting):引用計(jì)數(shù)機(jī)制。采用引用計(jì)數(shù)來(lái)管理對(duì)象的內(nèi)存,當(dāng)需要持有一個(gè)對(duì)象時(shí),使它的引用計(jì)數(shù) +1;當(dāng)不需要持有一個(gè)對(duì)象的時(shí)候,使它的引用計(jì)數(shù) -1;當(dāng)一個(gè)對(duì)象的引用計(jì)數(shù)為 0,該對(duì)象就會(huì)被銷(xiāo)毀。

Objective-C支持三種內(nèi)存管理機(jī)制:ARCMRCGC,但Objective-CGC機(jī)制有平臺(tái)局限性,僅限于MacOS開(kāi)發(fā)中,iOS開(kāi)發(fā)用的是RC機(jī)制,從MRC到現(xiàn)在的ARC

一個(gè)新創(chuàng)建的OC對(duì)象引用計(jì)數(shù)默認(rèn)是1,當(dāng)引用計(jì)數(shù)減為0,OC對(duì)象就會(huì)銷(xiāo)毀,釋放其占用的內(nèi)存空間

調(diào)用retain會(huì)讓OC對(duì)象的引用計(jì)數(shù)+1,調(diào)用release會(huì)讓OC對(duì)象的引用計(jì)數(shù)-1

內(nèi)存管理的經(jīng)驗(yàn)總結(jié)

  • 當(dāng)調(diào)用alloc、newcopy、mutableCopy方法返回了一個(gè)對(duì)象,在不需要這個(gè)對(duì)象時(shí),要調(diào)用release或者autorelease來(lái)釋放它
  • 想擁有某個(gè)對(duì)象,就讓它的引用計(jì)數(shù)+1;不想再擁有某個(gè)對(duì)象,就讓它的引用計(jì)數(shù)-1
  • 可以通過(guò)以下私有函數(shù)來(lái)查看自動(dòng)釋放池的情況

extern void _objc_autoreleasePoolPrint(void);

以上我們對(duì) “引用計(jì)數(shù)” 這一概念做了初步了解,Objective-C 中的 “對(duì)象” 通過(guò)引用計(jì)數(shù)功能來(lái)管理它的內(nèi)存生命周期。那么,對(duì)象的引用計(jì)數(shù)是如何存儲(chǔ)的呢?它存儲(chǔ)在哪個(gè)數(shù)據(jù)結(jié)構(gòu)里?

首先,不得不提一下isa。

isa

  • isa指針用來(lái)維護(hù) “對(duì)象” 和 “類(lèi)” 之間的關(guān)系,并確保對(duì)象和類(lèi)能夠通過(guò)isa指針找到對(duì)應(yīng)的方法、實(shí)例變量、屬性、協(xié)議等;
  • 在 arm64 架構(gòu)之前,isa就是一個(gè)普通的指針,直接指向objc_class,存儲(chǔ)著Class、Meta-Class對(duì)象的內(nèi)存地址。instance對(duì)象的isa指向class對(duì)象,class對(duì)象的isa指向meta-class對(duì)象;
  • 從 arm64 架構(gòu)開(kāi)始,對(duì)isa進(jìn)行了優(yōu)化,用nonpointer表示,變成了一個(gè)共用體(union)結(jié)構(gòu),還使用位域來(lái)存儲(chǔ)更多的信息。將 64 位的內(nèi)存數(shù)據(jù)分開(kāi)來(lái)存儲(chǔ)著很多的東西,其中的 33 位才是拿來(lái)存儲(chǔ)class、meta-class對(duì)象的內(nèi)存地址信息。要通過(guò)位運(yùn)算將isa的值& ISA_MASK掩碼,才能得到class、meta-class對(duì)象的內(nèi)存地址。
// objc.h
struct objc_object {
    Class isa;  // 在 arm64 架構(gòu)之前
};
// objc-private.h
struct objc_object {
private:
    isa_t isa;  // 在 arm64 架構(gòu)開(kāi)始
};
union isa_t 
{
    isa_t() { }
    isa_t(uintptr_t value) : bits(value) { }
    Class cls;
    uintptr_t bits;
#if SUPPORT_PACKED_ISA
    // extra_rc must be the MSB-most field (so it matches carry/overflow flags)
    // nonpointer must be the LSB (fixme or get rid of it)
    // shiftcls must occupy the same bits that a real class pointer would
    // bits + RC_ONE is equivalent to extra_rc + 1
    // RC_HALF is the high bit of extra_rc (i.e. half of its range)
    // future expansion:
    // uintptr_t fast_rr : 1;     // no r/r overrides
    // uintptr_t lock : 2;        // lock for atomic property, @synch
    // uintptr_t extraBytes : 1;  // allocated with extra bytes
# if __arm64__  // 在 __arm64__ 架構(gòu)下
#   define ISA_MASK        0x0000000ffffffff8ULL  // 用來(lái)取出 Class、Meta-Class 對(duì)象的內(nèi)存地址
#   define ISA_MAGIC_MASK  0x000003f000000001ULL
#   define ISA_MAGIC_VALUE 0x000001a000000001ULL
    struct {
        uintptr_t nonpointer        : 1;  // 0:代表普通的指針,存儲(chǔ)著 Class、Meta-Class 對(duì)象的內(nèi)存地址
                                          // 1:代表優(yōu)化過(guò),使用位域存儲(chǔ)更多的信息
        uintptr_t has_assoc         : 1;  // 是否有設(shè)置過(guò)關(guān)聯(lián)對(duì)象,如果沒(méi)有,釋放時(shí)會(huì)更快
        uintptr_t has_cxx_dtor      : 1;  // 是否有C++的析構(gòu)函數(shù)(.cxx_destruct),如果沒(méi)有,釋放時(shí)會(huì)更快
        uintptr_t shiftcls          : 33; // 存儲(chǔ)著 Class、Meta-Class 對(duì)象的內(nèi)存地址信息
        uintptr_t magic             : 6;  // 用于在調(diào)試時(shí)分辨對(duì)象是否未完成初始化
        uintptr_t weakly_referenced : 1;  // 是否有被弱引用指向過(guò),如果沒(méi)有,釋放時(shí)會(huì)更快
        uintptr_t deallocating      : 1;  // 對(duì)象是否正在釋放
        uintptr_t has_sidetable_rc  : 1;  // 如果為1,代表引用計(jì)數(shù)過(guò)大無(wú)法存儲(chǔ)在 isa 中,那么超出的引用計(jì)數(shù)會(huì)存儲(chǔ)在一個(gè)叫 SideTable 結(jié)構(gòu)體的 RefCountMap(引用計(jì)數(shù)表)散列表中
        uintptr_t extra_rc          : 19; // 里面存儲(chǔ)的值是對(duì)象本身之外的引用計(jì)數(shù)的數(shù)量,retainCount - 1
#       define RC_ONE   (1ULL<<45)
#       define RC_HALF  (1ULL<<18)
    };
......  // 在 __x86_64__ 架構(gòu)下
};

如果isanonpointer,即 arm64 架構(gòu)之前的isa指針。由于它只是一個(gè)普通的指針,存儲(chǔ)著ClassMeta-Class對(duì)象的內(nèi)存地址,所以它本身不能存儲(chǔ)引用計(jì)數(shù),所以以前對(duì)象的引用計(jì)數(shù)都存儲(chǔ)在一個(gè)叫SideTable結(jié)構(gòu)體的RefCountMap(引用計(jì)數(shù)表)散列表中。

如果isanonpointer,則它本身可以存儲(chǔ)一些引用計(jì)數(shù)。從以上union isa_t的定義中我們可以得知,isa_t中存儲(chǔ)了兩個(gè)引用計(jì)數(shù)相關(guān)的東西:extra_rchas_sidetable_rc

  • extra_rc:里面存儲(chǔ)的值是對(duì)象本身之外的引用計(jì)數(shù)的數(shù)量,這 19 位如果不夠存儲(chǔ),has_sidetable_rc的值就會(huì)變?yōu)?1;
  • has_sidetable_rc:如果為 1,代表引用計(jì)數(shù)過(guò)大無(wú)法存儲(chǔ)在isa中,那么超出的引用計(jì)數(shù)會(huì)存儲(chǔ)SideTableRefCountMap中。

所以,如果isanonpointer,則對(duì)象的引用計(jì)數(shù)存儲(chǔ)在它的isa_textra_rc中以及SideTableRefCountMap中。

SideTable

// NSObject.mm
struct SideTable {
    spinlock_t slock;        // 自旋鎖
    RefcountMap refcnts;     // 引用計(jì)數(shù)表(散列表)
    weak_table_t weak_table; // 弱引用表(散列表)
    ......
}

SideTable存儲(chǔ)在SideTables()中,SideTables()本質(zhì)也是一個(gè)散列表,可以通過(guò)對(duì)象指針來(lái)獲取它對(duì)應(yīng)的(引用計(jì)數(shù)表或者弱引用表)在哪一個(gè)SideTable中。在非嵌入式系統(tǒng)下,SideTables()中有 64 個(gè)SideTable。以下是SideTables()的定義:

// NSObject.mm
static objc::ExplicitInit<StripedMap<SideTable>> SideTablesMap;
static StripedMap<SideTable>& SideTables() {
    return SideTablesMap.get();
}

所以,查找對(duì)象的引用計(jì)數(shù)表需要經(jīng)過(guò)兩次哈希查找:

  • ① 第一次根據(jù)當(dāng)前對(duì)象的內(nèi)存地址,經(jīng)過(guò)哈希查找從SideTables()中取出它所在的SideTable;
  • ② 第二次根據(jù)當(dāng)前對(duì)象的內(nèi)存地址,經(jīng)過(guò)哈希查找從SideTable中的refcnts中取出它的引用計(jì)數(shù)表。

使用多個(gè)SideTable+分離鎖技術(shù)方案是為了保證線程安全的同時(shí)兼顧訪問(wèn)效率

以上就是iOS內(nèi)存管理引用計(jì)數(shù)示例分析的詳細(xì)內(nèi)容,更多關(guān)于iOS內(nèi)存管理引用計(jì)數(shù)的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

最新評(píng)論