1栗精、類的結(jié)構(gòu)
struct objc_class {
Class _Nonnull isa OBJC_ISA_AVAILABILITY;
#if !__OBJC2__
Class _Nullable super_class OBJC2_UNAVAILABLE;
const char * _Nonnull name OBJC2_UNAVAILABLE;
long version OBJC2_UNAVAILABLE;
long info OBJC2_UNAVAILABLE;
long instance_size OBJC2_UNAVAILABLE;
struct objc_ivar_list * _Nullable ivars OBJC2_UNAVAILABLE;
struct objc_method_list * _Nullable * _Nullable methodLists OBJC2_UNAVAILABLE;
struct objc_cache * _Nonnull cache OBJC2_UNAVAILABLE;
struct objc_protocol_list * _Nullable protocols OBJC2_UNAVAILABLE;
#endif
} OBJC2_UNAVAILABLE;
/* Use `Class` instead of `struct objc_class *` */
對象和類
Objective-C 是一門面向?qū)ο蟮木幊陶Z言涝桅。
對象都是一個類的實例惊奇,對象都有一個名為 isa 的指針,指向該對象的類开仰。
類描述了一系列它的實例的特點拟枚,包括成員變量的列表,成員函數(shù)的列表等众弓。
對象都可以接受消息恩溅,而對象能夠接收的消息列表是保存在它所對應(yīng)的類中。
類和元類
類也是一個對象谓娃,是元類 (metaclass)的實列脚乡,這個類就是元類 (metaclass)。
元類保存了類方法的列表滨达。
當(dāng)一個類方法被調(diào)用時奶稠,元類會首先查找它本身是否有該類方法的實現(xiàn),如果沒有捡遍,則該元類會向它的父類查找該方法锌订,直到一直找到繼承鏈的頭。
元類 (metaclass) 也是一個對象稽莉,那么元類的 isa 指針又指向哪里呢瀑志?
為了設(shè)計上的完整,所有的元類的 isa 指針都會指向一個根元類 (root metaclass)污秆。
根元類 (root metaclass) 本身的 isa 指針指向自己劈猪,這樣就行成了一個閉環(huán)。
無法動態(tài)給對象增加成員變量
因為對象在內(nèi)存中的排布可以看成一個結(jié)構(gòu)體良拼,該結(jié)構(gòu)體的大小并不能動態(tài)變化战得。
所以無法在運行時動態(tài)給對象增加成員變量。
可以動態(tài)給對象增加方法
相對的庸推,對象的方法的定義列表是一個名為 methodLists的指針的指針常侦。
通過修改該指針指向的指針的值,就可以實現(xiàn)動態(tài)地為某一個類增加成員方法贬媒。
這也是Category實現(xiàn)的原理聋亡。同時也說明了為什么Category只可為對象增加成員方法,卻不能增加成員變量际乘。
關(guān)聯(lián)對象
通過objc_setAssociatedObject 和 objc_getAssociatedObject方法可以變相地給對象增加成員變量坡倔,但由于實現(xiàn)機(jī)制不一樣,所以并不是真正改變了對象的內(nèi)存結(jié)構(gòu)。
2罪塔、isa-swizzling
就是把當(dāng)前某個實例對象的isa指針指向一個新建造的中間類投蝉,在這個新建造的中間類上面做hook方法或者別的事情,這樣不會影響這個類的其他實例對象征堪,僅僅影響當(dāng)前的實例對象瘩缆。
.class 和 object_getClass 的區(qū)分
.class 當(dāng) target 是 Instance 則返回 Class,當(dāng) target 是 Class 則返回自身
object_getClass 返回 isa 指針的指向
動態(tài)創(chuàng)建一個 Class 的完整步驟
objc_allocateClassPair
class_addMethod
class_addIvar
objc_registerClassPair
3佃蚜、isa-swizzling的應(yīng)用
KVO的實現(xiàn)
當(dāng)某個類的屬性對象第一次被觀察時庸娱,系統(tǒng)就會在運行期動態(tài)地創(chuàng)建該類的一個派生類,在這個派生類中重寫基類中任何被觀察屬性的setter 方法爽锥。
派生類在被重寫的setter方法內(nèi)實現(xiàn)真正的通知機(jī)制
如果原類為涌韩,那么生成的派生類名為NSKVONotifying_xxx
每個類對象中都有一個isa指針指向當(dāng)前類,當(dāng)一個類對象的第一次被觀察氯夷,那么系統(tǒng)會將isa指針指向動態(tài)生成的派生類臣樱,從而在給被監(jiān)控屬性賦值時執(zhí)行的是派生類的setter方法
鍵值觀察通知依賴于NSObject 的兩個方法:willChangeValueForKey: 和 didChangevlueForKey:;
在一個被觀察屬性發(fā)生改變之前腮考, willChangeValueForKey:一定會被調(diào)用雇毫,這就會記錄舊的值。
而當(dāng)改變發(fā)生后踩蔚,didChangeValueForKey:會被調(diào)用棚放,
繼而 observeValueForKey:ofObject:change:context: 也會被調(diào)用。
KVO的這套實現(xiàn)機(jī)制中蘋果重寫了class方法馅闽,讓我們誤認(rèn)為還是使用的當(dāng)前類飘蚯,從而達(dá)到隱藏生成的派生類
aspect AOP 面向切面編程的實現(xiàn)
aspect hook實例對象方法和類方法時候也是應(yīng)用了isa-swizzling,建造了新的派生類福也,在派生類上門進(jìn)行hook局骤,這樣移除hook的時候非常方便。