- 分類
- 擴展
- 代理(Delegate)
- 通知(NSNotification)
- KVO (Key-value observing)
- KVC (Key-value coding)
- 屬性關(guān)鍵字
一、分類
-
1.分類的作用撑蒜?
聲明私有方法歹啼,分解體積大的類文件,把framework的私有方法公開 -
2.分類的特點
運行時決議座菠,可以為系統(tǒng)類添加分類 狸眼。
說得詳細(xì)些,在運行時時期浴滴,將 Category 中的實例方法列表拓萌、協(xié)議列表、屬性列表添加到主類中后(所以Category中的方法在方法列表中的位置是在主類的同名方法之前的)升略,然后會遞歸調(diào)用所有類的 load 方法微王,這一切都是在main函數(shù)之前執(zhí)行的。 -
3.分類可以添加哪些內(nèi)容品嚣?
實例方法炕倘,類方法,協(xié)議腰根,屬性(添加getter和setter方法激才,并沒有實例變量,添加實例變量需要用關(guān)聯(lián)對象) -
4.如果工程里有兩個分類A和B,兩個分類中有一個同名的方法瘸恼,哪個方法最終生效劣挫?
取決于分類的編譯順序,最后編譯的那個分類的同名方法最終生效东帅,而之前的都會被覆蓋掉(這里并不是真正的覆蓋压固,因為其余方法仍然存在,只是訪問不到靠闭,因為在動態(tài)添加類的方法的時候是倒序遍歷方法列表的帐我,而最后編譯的分類的方法會放在方法列表前面,訪問的時候就會先被訪問到愧膀,同理如果聲明了一個和原類方法同名的方法拦键,也會覆蓋掉原類的方法)。 -
5.如果聲明了兩個同名的分類會怎樣檩淋?
會報錯芬为,所以第三方的分類,一般都帶有命名前綴 -
6.分類能添加成員變量嗎蟀悦?
不能媚朦。只能通過關(guān)聯(lián)對象(objc_setAssociatedObject)來模擬實現(xiàn)成員變量,但其實質(zhì)是關(guān)聯(lián)內(nèi)容日戈,所有對象的關(guān)聯(lián)內(nèi)容都放在同一個全局容器哈希表中:AssociationsHashMap,由AssociationsManager統(tǒng)一管理询张。
二、擴展
-
1.一般用擴展做什么浙炼?
聲明私有屬性份氧,聲明方法(沒什么意義),聲明私有成員變量 -
2.擴展的特點
編譯時決議鼓拧,只能以聲明的形式存在半火,多數(shù)情況下寄生在宿主類的.m中,不能為系統(tǒng)類添加擴展季俩。
三钮糖、代理(Delegate)
代理是一種設(shè)計模式,以@protocol形式體現(xiàn)酌住,一般是一對一傳遞店归。
一般以weak關(guān)鍵詞以規(guī)避循環(huán)引用。
四酪我、通知(NSNotification)
使用觀察者模式來實現(xiàn)的用于跨層傳遞信息的機制消痛。傳遞方式是一對多的。
-
如果實現(xiàn)通知機制都哭?
image
五秩伞、KVO (Key-value observing)
KVO是觀察者模式的另一實現(xiàn)逞带。
使用了isa混寫(isa-swizzling)來實現(xiàn)KVO
使用setter方法改變值KVO會生效,使用setValue:forKey即KVC改變值KVO也會生效纱新,因為KVC會去調(diào)用setter方法
- (void)setValue:(id)value
{
[self willChangeValueForKey:@"key"];
[super setValue:value];
[self didChangeValueForKey:@"key"];
}
-
那么通過直接賦值成員變量會觸發(fā)KVO嗎展氓?
不會,因為不會調(diào)用setter方法脸爱,需要加上
willChangeValueForKey和didChangeValueForKey方法來手動觸發(fā)才行
六遇汞、KVC(Key-value coding)
-(id)valueForKey:(NSString *)key;
-(void)setValue:(id)value forKey:(NSString *)key;
KVC就是指iOS的開發(fā)中,可以允許開發(fā)者通過Key名直接訪問對象的屬性簿废,或者給對象的屬性賦值空入。而不需要調(diào)用明確的存取方法。這樣就可以在運行時動態(tài)地訪問和修改對象的屬性族檬。而不是在編譯時確定歪赢,這也是iOS開發(fā)中的黑魔法之一。很多高級的iOS開發(fā)技巧都是基于KVC實現(xiàn)的
當(dāng)調(diào)用setValue:屬性值 forKey:@”name“的代碼時导梆,轨淌,底層的執(zhí)行機制如下:
- 程序優(yōu)先調(diào)用set<Key>:屬性值方法,代碼通過setter方法完成設(shè)置看尼。注意,這里的<key>是指成員變量名盟步,首字母大小寫要符合KVC的命名規(guī)則藏斩,下同
- 如果沒有找到setName:方法,KVC機制會檢查+ (BOOL)accessInstanceVariablesDirectly方法有沒有返回YES却盘,默認(rèn)該方法會返回YES狰域,如果你重寫了該方法讓其返回NO的話,那么在這一步KVC會執(zhí)行setValue:forUndefinedKey:方法黄橘,不過一般開發(fā)者不會這么做兆览。所以KVC機制會搜索該類里面有沒有名為<key>的成員變量,無論該變量是在類接口處定義塞关,還是在類實現(xiàn)處定義抬探,也無論用了什么樣的訪問修飾符,只在存在以<key>命名的變量帆赢,KVC都可以對該成員變量賦值小压。
- 如果該類即沒有set<key>:方法,也沒有_<key>成員變量椰于,KVC機制會搜索_is<Key>的成員變量怠益。
- 和上面一樣,如果該類即沒有set<Key>:方法瘾婿,也沒有_<key>和_is<Key>成員變量蜻牢,KVC機制再會繼續(xù)搜索<key>和is<Key>的成員變量烤咧。再給它們賦值。
- 如果上面列出的方法或者成員變量都不存在抢呆,系統(tǒng)將會執(zhí)行該對象的setValue:forUndefinedKey:方法髓削,默認(rèn)是拋出異常。
即如果沒有找到Set<Key>方法的話镀娶,會按照_key立膛,_iskey,key梯码,iskey的順序搜索成員并進行賦值操作宝泵。
如果開發(fā)者想讓這個類禁用KVC,那么重寫+ (BOOL)accessInstanceVariablesDirectly方法讓其返回NO即可轩娶,這樣的話如果KVC沒有找到set<Key>:屬性名時儿奶,會直接用setValue:forUndefinedKey:方法。
當(dāng)調(diào)用valueForKey:@”name“的代碼時鳄抒,KVC對key的搜索方式不同于setValue:屬性值 forKey:@”name“闯捎,其搜索方式如下:
- 首先按get<Key>,<key>,is<Key>的順序方法查找getter方法,找到的話會直接調(diào)用许溅。如果是BOOL或者Int等值類型瓤鼻, 會將其包裝成一個NSNumber對象。
- 如果上面的getter沒有找到贤重,KVC則會查找countOf<Key>,objectIn<Key>AtIndex或<Key>AtIndexes格式的方法茬祷。如果countOf<Key>方法和另外兩個方法中的一個被找到,那么就會返回一個可以響應(yīng)NSArray所有方法的代理集合(它是NSKeyValueArray并蝗,是NSArray的子類)祭犯,調(diào)用這個代理集合的方法,或者說給這個代理集合發(fā)送屬于NSArray的方法滚停,就會以countOf<Key>,objectIn<Key>AtIndex或<Key>AtIndexes這幾個方法組合的形式調(diào)用沃粗。還有一個可選的get<Key>:range:方法。所以你想重新定義KVC的一些功能键畴,你可以添加這些方法最盅,需要注意的是你的方法名要符合KVC的標(biāo)準(zhǔn)命名方法,包括方法簽名镰吵。
- 如果上面的方法沒有找到檩禾,那么會同時查找countOf<Key>,enumeratorOf<Key>,memberOf<Key>格式的方法疤祭。如果這三個方法都找到盼产,那么就返回一個可以響應(yīng)NSSet所的方法的代理集合,和上面一樣勺馆,給這個代理集合發(fā)NSSet的消息戏售,就會以countOf<Key>侨核,enumeratorOf<Key>,memberOf<Key>組合的形式調(diào)用。
- 如果還沒有找到灌灾,再檢查類方法+ (BOOL)accessInstanceVariablesDirectly,如果返回YES(默認(rèn)行為)搓译,那么和先前的設(shè)值一樣,會按_<key>,_is<Key>,<key>,is<Key>的順序搜索成員變量名锋喜,這里不推薦這么做些己,因為這樣直接訪問實例變量破壞了封裝性晨逝,使代碼更脆弱枝恋。如果重寫了類方法+ (BOOL)accessInstanceVariablesDirectly返回NO的話,那么會直接調(diào)用valueForUndefinedKey:方法可款,默認(rèn)是拋出異常炉奴。
七逼庞、屬性關(guān)鍵字
1.讀寫權(quán)限:readonly,readwrite(默認(rèn))
2.原子性: atomic(默認(rèn)),nonatomic瞻赶。atomic讀寫線程安全赛糟,但效率低,而且不是絕對的安全砸逊,比如如果修飾的是數(shù)組璧南,那么對數(shù)組的讀寫是安全的,但如果是操作數(shù)組進行添加移除其中對象的還痹兜,就不保證安全了穆咐。
3.引用計數(shù):
- retain/strong
- assign:修飾基本數(shù)據(jù)類型,修飾對象類型時字旭,不改變其引用計數(shù),會產(chǎn)生懸垂指針崖叫,修飾的對象在被釋放后遗淳,assign指針仍然指向原對象內(nèi)存地址,如果使用assign指針繼續(xù)訪問原對象的話心傀,就可能會導(dǎo)致內(nèi)存泄漏或程序異常
- weak:不改變被修飾對象的引用計數(shù)屈暗,所指對象在被釋放后,weak指針會自動置為nil
- copy:分為深拷貝和淺拷貝
淺拷貝:對內(nèi)存地址的復(fù)制脂男,讓目標(biāo)對象指針和原對象指向同一片內(nèi)存空間會增加引用計數(shù)
深拷貝:對對象內(nèi)容的復(fù)制养叛,開辟新的內(nèi)存空間
可變對象的copy和mutableCopy都是深拷貝
不可變對象的copy是淺拷貝,mutableCopy是深拷貝
copy方法返回的都是不可變對象
-
@property (nonatomic, copy) NSMutableArray * array;這樣寫有什么影響宰翅?
因為copy方法返回的都是不可變對象弃甥,所以array對象實際上是不可變的,如果對其進行可變操作如添加移除對象汁讼,則會造成程序crash