iOS面試題基礎

分類、擴展、代理、通知、KVC、KVO、屬性關鍵字

一、分類(Category)

1、分類的作用?

聲明私有方法,分解體積大的類文件

2、分類的特點?

可以爲系統類添加分類。在運行時時期,將 Category 中的實例方法列表、協議列表、屬性列表添加到主類中後(所有Category中的方法在方法列表中的位置是在主類的同名方法之前的),然後會遞歸調用所有類的 load 方法,這一切都是在main函數之前執行的。

3、分類可以添加哪些內容?

實例方法,類方法,屬性(添加getter和setter方法,並沒有實例變量,添加實例變量需要用關聯對象)

4、如果工程裏有兩個分類A和B,兩個分類中有一個同名的方法,哪個方法最終生效?

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

5、如果聲明瞭兩個同名的分類會怎樣?

會報錯,所以第三方的分類,一般都帶有命名前綴

6、分類能添加成員變量嗎?

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

二、擴展(Extension)

1、擴展的作用?

聲明私有屬性,聲明私有成員變量

2、擴展的特點?

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

三、代理(Delegate)

代理是一種設計模式,委託方聲明協議,定義需要實現的接口,代理方按照協議實現方法

一般用weak來避免循環引用

四、通知(NSNotification)

使用觀察者模式用於實現跨層傳遞信息的機制。傳遞方式是一對多。

五、KVO(key-value-observing)

KVO是觀察者的另一實現

使用了isa混寫(isa-swizzling)來實現KVO

使用setter方法改變值KVO會生效,使用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)

KVC可以通過key直接訪問對象的屬性,或者給對象的屬性賦值,這樣可以在運行時動態的訪問或修改對象的屬性

當調用setValue:屬性值 forKey:@”name“的代碼時,,底層的執行機制如下:

1、程序優先調用set<Key>:屬性值方法,代碼通過setter方法完成設置。注意,這裏的<key>是指成員變量名,首字母大小寫要符合KVC的命名規則,下同

2、如果沒有找到setName:方法,KVC機制會檢查+ (BOOL)accessInstanceVariablesDirectly方法有沒有返回YES,默認該方法會返回YES,如果你重寫了該方法讓其返回NO的話,那麼在這一步KVC會執行setValue:forUndefinedKey:方法,不過一般開發者不會這麼做。所以KVC機制會搜索該類裏面有沒有名爲<key>的成員變量,無論該變量是在類接口處定義,還是在類實現處定義,也無論用了什麼樣的訪問修飾符,只在存在以<key>命名的變量,KVC都可以對該成員變量賦值。

3、如果該類即沒有set<key>:方法,也沒有_<key>成員變量,KVC機制會搜索_is<Key>的成員變量。

4、和上面一樣,如果該類即沒有set<Key>:方法,也沒有_<key>和_is<Key>成員變量,KVC機制再會繼續搜索<key>和is<Key>的成員變量。再給它們賦值。

5、如果上面列出的方法或者成員變量都不存在,系統將會執行該對象的setValue:forUndefinedKey:方法,默認是拋出異常。

如果想禁用KVC,重寫+ (BOOL)accessInstanceVariablesDirectly方法讓其返回NO即可,這樣的話如果KVC沒有找到set<Key>:屬性名時,會直接用setValue:forUndefinedKey:方法。

當調用valueForKey:@”name“的代碼時,KVC對key的搜索方式不同於setValue:屬性值 forKey:@”name“,其搜索方式如下:

1、首先按get<Key>,<key>,is<Key>的順序方法查找getter方法,找到的話會直接調用。如果是BOOL或者Int等值類型, 會將其包裝成一個NSNumber對象

2、如果上面的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的標準命名方法,包括方法簽名。

3、如果上面的方法沒有找到,那麼會同時查找countOf<Key>,enumeratorOf<Key>,memberOf<Key>格式的方法。如果這三個方法都找到,那麼就返回一個可以響應NSSet所的方法的代理集合,和上面一樣,給這個代理集合發NSSet的消息,就會以countOf<Key>,enumeratorOf<Key>,memberOf<Key>組合的形式調用。

4、如果還沒有找到,再檢查類方法+ (BOOL)accessInstanceVariablesDirectly,如果返回YES(默認行爲),那麼和先前的設值一樣,會按_<key>,_is<Key>,<key>,is<Key>的順序搜索成員變量名,這裏不推薦這麼做,因爲這樣直接訪問實例變量破壞了封裝性,使代碼更脆弱。如果重寫了類方法+ (BOOL)accessInstanceVariablesDirectly返回NO的話,那麼會直接調用valueForUndefinedKey:方法,默認是拋出異常

七、屬性關鍵字

1、讀寫權限:readonly,readwrite(默認)

2、原子性:(atomic),nonatimic。atomic讀寫安全,但效率低,不是絕對的安全,比如操作數組,增加或移除,這種情況可以使用互斥鎖來保證線程安全

3、引用計數

retain/strong

assign修飾基本數據類型

weak不改變修飾對象的引用計數,對象釋放後,weak指針自動置爲空

copy分深拷貝和淺拷貝

淺拷貝,對象指針的複製,目標對象指針和源對象指針指向同一塊內存空間,引用計數增加

深拷貝,對象內容的複製,開闢一塊新的內存空間

可變的對象的copy和mutableCopy都是深拷貝

不可變對象的copy是淺拷貝,mutable是深拷貝

copy方法返回的都是不可變對象

@property (nonatomic, copy) NSMutableArray * array;這樣使用會crash,因爲copy的對象是不可變的

NSString使用copy修飾不用strong修飾,用strong修飾一個name屬性,如果賦值的是一個可變對象,當可變對象的值發生改變的時候,name的值也會改變,這不是我們期望的,是因爲name使用strong修飾後,指向跟可變對象相同的一塊內存地址,如果使用copy的話,則是深拷貝,會開闢一塊新的內存空間,因此可變對象值變化時,也不會影響name的值。

 

 

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章