- alloc
alloc 經(jīng)過一系列調(diào)用最終調(diào)用的c函數(shù)中的calloc方法如绸,在alloc對象時(shí)完域,引用計(jì)數(shù)并沒有加1.那為什么在使用retainCount獲取時(shí)软吐,引用計(jì)數(shù)+1 了呢?吟税?
2.retain
在使用retain時(shí)凹耙,系統(tǒng)是怎樣查找引用計(jì)數(shù)的?肠仪?
兩個(gè)Hash查找肖抱,
第一次查找到:在那一張sideTable 表中,
對象指針通過相應(yīng)的Hash函數(shù)獲取到相應(yīng)的存儲ke y异旧,即在SideTables 的索引位置虐沥。從而獲取到相應(yīng)的SideTable 表。
第二次:查找到:在sideTable 表中的具體位置泽艘。
通過當(dāng)前變量的引用計(jì)數(shù)指針到SideTable 中欲险,獲取引用計(jì)數(shù)Map.即它的引用計(jì)數(shù)size_t.然后進(jìn)行加1操作,所謂的+1匹涮,其實(shí)是偏移量天试。
- release的相關(guān)實(shí)現(xiàn)原理
- retainCount 實(shí)現(xiàn)原理
首先查找到對應(yīng)的SideTale ,然后,聲明一個(gè)變量
size_t refcnt_result = 1;
最后到SideTable中查找然低,找到對應(yīng)的你用計(jì)數(shù)size_t ,再進(jìn)行偏移喜每,偏移后的值,在與refcnt_result 進(jìn)行+= 操作雳攘。 由于新alloc的對象沒有key-value的映射的带兜,所以獲取到的值是0.經(jīng)過retainCount 的中的+= 操作,最后成為了1.
5.dealloc 的實(shí)現(xiàn)原理
開始--> 首先會調(diào)用_objc_rootDealloc()的私有函數(shù)-->私有函數(shù)中吨灭,又會調(diào)用rootDealloc 函數(shù)--> 然后在其內(nèi)部刚照,會判斷,當(dāng)前對象是否可以直接釋放
釋放的判斷條件:
(1)nonpointer_isa 是否使用了非指針型isa.
(2)是否有弱引用關(guān)系
(3)是否有關(guān)聯(lián)對象
(4)是否有c++或ARC相關(guān)代碼
(5)是否有掛載的散列表
全都沒有的情況下喧兄,可以直接調(diào)用c函數(shù)的free進(jìn)行釋放无畔,否則需要調(diào)用objc_dispose()函數(shù),清除所有關(guān)系后吠冤,再進(jìn)行釋放的操作浑彰。
objc_dispose 的內(nèi)部實(shí)現(xiàn)?拯辙?
首先會調(diào)用objc_destructInstance()郭变,在這個(gè)方法中,首先判斷是否有c++或者ARC相關(guān)內(nèi)容,如果說存在就去調(diào)用objc_cxxDestruct()銷毀诉濒,如果說沒有周伦,就會判斷是否存在關(guān)聯(lián)對象,如果存在關(guān)聯(lián)對象的話循诉,就去調(diào)用_object_reomve_assocations() 方法進(jìn)行移除,如果說沒有撇他,系統(tǒng)會調(diào)用clearDeallocating()函數(shù)茄猫,在該函數(shù)中,首先調(diào)用了一個(gè)sideTable_clearDeallcating()函數(shù)困肩,之后回到用weak_clear_no_lock()函數(shù)划纽,該函數(shù)是將指向該對象的弱引用指針置為 nil 這個(gè)方法過后,會調(diào)用table.refcnts.erase()引用計(jì)數(shù)擦出即從引用計(jì)數(shù)表中锌畸,將該對象的引用計(jì)數(shù)擦出勇劣。.然后再調(diào)用C函數(shù)free結(jié)束實(shí)現(xiàn)。