iOS面試題---Objective_C語言特性:分類、擴展、代理、通知、KVO、KVC、屬性

更多的文章請看-2020iOS面試大全 持續更新!

  • 分類
  • 擴展
  • 代理(Delegate)
  • 通知(NSNotification)
  • KVO (Key-value observing)
  • KVC (Key-value coding)
  • 屬性關鍵字
 

1、分類web

  • 1.分類的做用?

    聲明私有方法,分解體積大的類文件,把framework的私有方法公開
  • 2.分類的特色

    運行時決議,能夠爲系統類添加分類 。
    說得詳細些,在運行時時期,將 Category 中的實例方法列表、協議列表、屬性列表添加到主類中後(因此Category中的方法在方法列表中的位置是在主類的同名方法以前的),而後會遞歸調用全部類的 load 方法,這一切都是在main函數以前執行的。
  • 3.分類能夠添加哪些內容?

    實例方法,類方法,協議,屬性(添加getter和setter方法,並無實例變量,添加實例變量須要用關聯對象)
  • 4.若是工程裏有兩個分類A和B,兩個分類中有一個同名的方法,哪一個方法最終生效?

    取決於分類的編譯順序,最後編譯的那個分類的同名方法最終生效,而以前的都會被覆蓋掉(這裏並非真正的覆蓋,由於其他方法仍然存在,只是訪問不到,由於在動態添加類的方法的時候是倒序遍歷方法列表的,而最後編譯的分類的方法會放在方法列表前面,訪問的時候就會先被訪問到,同理若是聲明瞭一個和原類方法同名的方法,也會覆蓋掉原類的方法)。
  • 5.若是聲明瞭兩個同名的分類會怎樣?

    會報錯,因此第三方的分類,通常都帶有命名前綴
  • 6.分類能添加成員變量嗎?

    不能。只能經過關聯對象(objc_setAssociatedObject)來模擬實現成員變量,但其實質是關聯內容,全部對象的關聯內容都放在同一個全局容器哈希表中:AssociationsHashMap,由AssociationsManager統一管理。

學習的圈子特別重要!!

附上一份收集的各大廠面試題(附答案) ! 羣文件直接獲取
各大廠面試題
面試

做爲一個開發者,有一個學習的氛圍跟一個交流圈子特別重要,這是一個個人iOS交流羣:761407670 進羣密碼‘博客’,無論你是小白仍是大牛歡迎入駐 ,分享BAT,阿里面試題、面試經驗,討論技術, 你們一塊兒交流學習成長!設計模式

 

 

2、擴展數組

  • 1.通常用擴展作什麼?

    聲明私有屬性,聲明方法(沒什麼意義),聲明私有成員變量
  • 2.擴展的特色

    編譯時決議,只能以聲明的形式存在,多數狀況下寄生在宿主類的.m中,不能爲系統類添加擴展。

3、代理(Delegate)安全

 
image.png

代理是一種設計模式,以@protocol形式體現,通常是一對一傳遞。
通常以weak關鍵詞以規避循環引用。bash

4、通知(NSNotification)
使用觀察者模式來實現的用於跨層傳遞信息的機制。傳遞方式是一對多的。函數

  • 若是實現通知機制?

     
    image.png

5、KVO (Key-value observing)
KVO是觀察者模式的另外一實現。
使用了isa混寫(isa-swizzling)來實現KVO學習

 
image.png

使用setter方法改變值KVO會生效,使用setValue:forKey即KVC改變值KVO也會生效,由於KVC會去調用setter方法ui

 
- (void)setValue:(id)value { [self willChangeValueForKey:@"key"]; [super setValue:value]; [self didChangeValueForKey:@"key"]; } 複製代碼
  • 那麼經過直接賦值成員變量會觸發KVO嗎?

    不會,由於不會調用setter方法,須要加上
    willChangeValueForKey和didChangeValueForKey方法來手動觸發才行

6、KVC(Key-value coding)atom

 
-(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:方法,默認是拋出異常。

7、屬性關鍵字
1.讀寫權限:readonly,readwrite(默認)
2.原子性: atomic(默認),nonatomic。atomic讀寫線程安全,但效率低,並且不是絕對的安全,好比若是修飾的是數組,那麼對數組的讀寫是安全的,但若是是操做數組進行添加移除其中對象的還,就不保證安全了。
3.引用計數:

  • retain/strong
  • assign:修飾基本數據類型,修飾對象類型時,不改變其引用計數,會產生懸垂指針,修飾的對象在被釋放後,assign指針仍然指向原對象內存地址,若是使用assign指針繼續訪問原對象的話,就可能會致使內存泄漏或程序異常
  • weak:不改變被修飾對象的引用計數,所指對象在被釋放後,weak指針會自動置爲nil
  • copy:分爲深拷貝和淺拷貝
    淺拷貝:對內存地址的複製,讓目標對象指針和原對象指向同一片內存空間會增長引用計數
    深拷貝:對對象內容的複製,開闢新的內存空間
 
image.png

可變對象的copy和mutableCopy都是深拷貝
不可變對象的copy是淺拷貝,mutableCopy是深拷貝
copy方法返回的都是不可變對象

  • @property (nonatomic, copy) NSMutableArray * array;這樣寫有什麼影響?

    由於copy方法返回的都是不可變對象,因此array對象其實是不可變的,若是對其進行可變操做如添加移除對象,則會形成程序crash

更多的文章請看-2020iOS面試大全 持續更新!

學習的圈子特別重要!!

附上一份收集的各大廠面試題(附答案) ! 羣文件直接獲取
各大廠面試題

做爲一個開發者,有一個學習的氛圍跟一個交流圈子特別重要,這是一個個人iOS交流羣:761407670 進羣密碼‘博客’,無論你是小白仍是大牛歡迎入駐 ,分享BAT,阿里面試題、面試經驗,討論技術, 你們一塊兒交流學習成長!

相關文章
相關標籤/搜索