更多的文章請看-2020iOS面試大全 持續更新!
- 分類
- 擴展
- 代理(Delegate)
- 通知(NSNotification)
- KVO (Key-value observing)
- KVC (Key-value coding)
- 屬性關鍵字
一、分類
-
1.分類的作用?
宣告私有方法,分解體積大的類檔案,把framework的私有方法公開 -
2.分類的特點
運行時決議,可以為系統類添加分類 ,
說得詳細些,在運行時時期,將 Category 中的實體方法串列、協議串列、屬性串列添加到主類中后(所以Category中的方法在方法串列中的位置是在主類的同名方法之前的),然后會遞回呼叫所有類的 load 方法,這一切都是在main函式之前執行的, -
3.分類可以添加哪些內容?
實體方法,類方法,協議,屬性(添加getter和setter方法,并沒有實體變數,添加實體變數需要用關聯物件) -
4.如果工程里有兩個分類A和B,兩個分類中有一個同名的方法,哪個方法最終生效?
取決于分類的編譯順序,最后編譯的那個分類的同名方法最終生效,而之前的都會被覆寫掉(這里并不是真正的覆寫,因為其余方法仍然存在,只是訪問不到,因為在動態添加類的方法的時候是倒序遍歷方法串列的,而最后編譯的分類的方法會放在方法串列前面,訪問的時候就會先被訪問到,同理如果宣告了一個和原類方法同名的方法,也會覆寫掉原類的方法), -
5.如果宣告了兩個同名的分類會怎樣?
會報錯,所以第三方的分類,一般都帶有命名前綴 -
6.分類能添加成員變數嗎?
不能,只能通過關聯物件(objc_setAssociatedObject)來模擬實作成員變數,但其實質是關聯內容,所有物件的關聯內容都放在同一個全域容器哈希表中:AssociationsHashMap,由AssociationsManager統一管理,
學習的圈子特別重要!!
附上一份收集的各大廠面試題(附答案) ! 群檔案直接獲取
各大廠面試題
作為一個開發者,有一個學習的氛圍跟一個交流圈子特別重要,這是一個我的iOS交流群:761407670 進群密碼‘博客’,不管你是小白還是大牛歡迎入駐 ,分享BAT,阿里面試題、面試經驗,討論技術, 大家一起交流學習成長!
二、擴展
-
1.一般用擴展做什么?
宣告私有屬性,宣告方法(沒什么意義),宣告私有成員變數 -
2.擴展的特點
編譯時決議,只能以宣告的形式存在,多數情況下寄生在宿主類的.m中,不能為系統類添加擴展,
三、代理(Delegate)
代理是一種設計模式,以@protocol形式體現,一般是一對一傳遞,
一般以weak關鍵詞以規避回圈參考,
四、通知(NSNotification)
使用觀察者模式來實作的用于跨層傳遞資訊的機制,傳遞方式是一對多的,
-
如果實作通知機制?
image.png
五、KVO (Key-value observing)
KVO是觀察者模式的另一實作,
使用了isa混寫(isa-swizzling)來實作KVO
使用setter方法改變值KVO會生效,使用setValue:forKey即KVC改變值KVO也會生效,因為KVC會去呼叫setter方法
- (void)setValue:(id)value
{
[self willChangeValueForKey:@"key"];
[super setValue:value];
[self didChangeValueForKey:@"key"];
}
復制代碼
-
那么通過直接賦值成員變數會觸發KVO嗎?
不會,因為不會呼叫setter方法,需要加上
willChangeValueForKey和didChangeValueForKey方法來手動觸發才行
六、KVC(Key-value coding)
-(id)valueForKey:(NSString *)key;
-(void)setValue:(id)value forKey:(NSString *)key;
復制代碼
KVC就是指iOS的開發中,可以允許開發者通過Key名直接訪問物件的屬性,或者給物件的屬性賦值,而不需要呼叫明確的存取方法,這樣就可以在運行時動態地訪問和修改物件的屬性,而不是在編譯時確定,這也是iOS開發中的黑魔法之一,很多高級的iOS開發技巧都是基于KVC實作的
當呼叫setValue:屬性值 forKey:@”name“的代碼時,,底層的執行機制如下:
- 程式優先呼叫set<Key>:屬性值方法,代碼通過setter方法完成設定,注意,這里的<key>是指成員變數名,首字母大小寫要符合KVC的命名規則,下同
- 如果沒有找到setName:方法,KVC機制會檢查+ (BOOL)accessInstanceVariablesDirectly方法有沒有回傳YES,默認該方法會回傳YES,如果你重寫了該方法讓其回傳NO的話,那么在這一步KVC會執行setValue:forUndefinedKey:方法,不過一般開發者不會這么做,所以KVC機制會搜索該類里面有沒有名為<key>的成員變數,無論該變數是在類介面處定義,還是在類實作處定義,也無論用了什么樣的訪問修飾符,只在存在以<key>命名的變數,KVC都可以對該成員變數賦值,
- 如果該類即沒有set<key>:方法,也沒有_<key>成員變數,KVC機制會搜索_is<Key>的成員變數,
- 和上面一樣,如果該類即沒有set<Key>:方法,也沒有_<key>和_is<Key>成員變數,KVC機制再會繼續搜索<key>和is<Key>的成員變數,再給它們賦值,
- 如果上面列出的方法或者成員變數都不存在,系統將會執行該物件的setValue:forUndefinedKey:方法,默認是拋出例外,
即如果沒有找到Set<Key>方法的話,會按照_key,_iskey,key,iskey的順序搜索成員并進行賦值操作,
如果開發者想讓這個類禁用KVC,那么重寫+ (BOOL)accessInstanceVariablesDirectly方法讓其回傳NO即可,這樣的話如果KVC沒有找到set<Key>:屬性名時,會直接用setValue:forUndefinedKey:方法,
當呼叫valueForKey:@”name“的代碼時,KVC對key的搜索方式不同于setValue:屬性值 forKey:@”name“,其搜索方式如下:
- 首先按get<Key>,<key>,is<Key>的順序方法查找getter方法,找到的話會直接呼叫,如果是BOOL或者Int等值型別, 會將其包裝成一個NSNumber物件,
- 如果上面的getter沒有找到,KVC則會查找countOf<Key>,objectIn<Key>AtIndex或<Key>AtIndexes格式的方法,如果countOf<Key>方法和另外兩個方法中的一個被找到,那么就會回傳一個可以回應NSArray所有方法的代理集合(它是NSKeyValueArray,是NSArray的子類),呼叫這個代理集合的方法,或者說給這個代理集合發送屬于NSArray的方法,就會以countOf<Key>,objectIn<Key>AtIndex或<Key>AtIndexes這幾個方法組合的形式呼叫,還有一個可選的get<Key>:range:方法,所以你想重新定義KVC的一些功能,你可以添加這些方法,需要注意的是你的方法名要符合KVC的標準命名方法,包括方法簽名,
- 如果上面的方法沒有找到,那么會同時查找countOf<Key>,enumeratorOf<Key>,memberOf<Key>格式的方法,如果這三個方法都找到,那么就回傳一個可以回應NSSet所的方法的代理集合,和上面一樣,給這個代理集合發NSSet的訊息,就會以countOf<Key>,enumeratorOf<Key>,memberOf<Key>組合的形式呼叫,
- 如果還沒有找到,再檢查類方法+ (BOOL)accessInstanceVariablesDirectly,如果回傳YES(默認行為),那么和先前的設值一樣,會按_<key>,_is<Key>,<key>,is<Key>的順序搜索成員變數名,這里不推薦這么做,因為這樣直接訪問實體變數破壞了封裝性,使代碼更脆弱,如果重寫了類方法+ (BOOL)accessInstanceVariablesDirectly回傳NO的話,那么會直接呼叫valueForUndefinedKey:方法,默認是拋出例外,
七、屬性關鍵字
1.讀寫權限:readonly,readwrite(默認)
2.原子性: atomic(默認),nonatomic,atomic讀寫執行緒安全,但效率低,而且不是絕對的安全,比如如果修飾的是陣列,那么對陣列的讀寫是安全的,但如果是操作陣列進行添加移除其中物件的還,就不保證安全了,
3.參考計數:
- retain/strong
- assign:修飾基本資料型別,修飾物件型別時,不改變其參考計數,會產生懸垂指標,修飾的物件在被釋放后,assign指標仍然指向原物件記憶體地址,如果使用assign指標繼續訪問原物件的話,就可能會導致記憶體泄漏或程式例外
- weak:不改變被修飾物件的參考計數,所指物件在被釋放后,weak指標會自動置為nil
- copy:分為深拷貝和淺拷貝
淺拷貝:對記憶體地址的復制,讓目標物件指標和原物件指向同一片記憶體空間會增加參考計數
深拷貝:對物件內容的復制,開辟新的記憶體空間
可變物件的copy和mutableCopy都是深拷貝
不可變物件的copy是淺拷貝,mutableCopy是深拷貝
copy方法回傳的都是不可變物件
-
@property (nonatomic, copy) NSMutableArray * array;這樣寫有什么影響?
因為copy方法回傳的都是不可變物件,所以array物件實際上是不可變的,如果對其進行可變操作如添加移除物件,則會造成程式crash
更多的文章請看-2020iOS面試大全 持續更新!
學習的圈子特別重要!!
附上一份收集的各大廠面試題(附答案) ! 群檔案直接獲取
各大廠面試題
作為一個開發者,有一個學習的氛圍跟一個交流圈子特別重要,這是一個我的iOS交流群:761407670 進群密碼‘博客’,不管你是小白還是大牛歡迎入駐 ,分享BAT,阿里面試題、面試經驗,討論技術, 大家一起交流學習成長!
轉載請註明出處,本文鏈接:https://www.uj5u.com/yidong/2556.html
標籤:iOS
