1、NSString能不能被繼承焰轻,為什么?
不可以妇多,先看一下NSString的alloc方法
@class NSPlaceholderString;
@interface NSString:(NSObject)
+ (id) alloc;
@ end
@implementation NSString
+(id) alloc {
if ([self isEquals:[NSString class]]) {
return [NSPlaceholderString alloc];
}
else
return [super alloc];
}
@end
@interface NSPlaceholderString:(NSString)
@end
在alloc方法中我們可以看到焕议,當(dāng)只用NSString調(diào)用alloc的時(shí)候,由于self == [NSString class]殿较,所以這時(shí)返回的是NSPlaceholderString的類對(duì)象;而使用其他類(比如派生類)調(diào)用alloc時(shí),返回的是super的 alloc笋额,這里也就是[NSObject alloc],而NSObject的alloc方法返回的是調(diào)用類的類對(duì)象拨齐,所以在我們用我們自己的XXString就是XXString類的類對(duì)象了所以沒(méi)有NSString 的一些方法了鳞陨。
2、如何給類如UILabel添加方法瞻惋,底層是什么
使用category對(duì)類添加方法厦滤,Objective-C 中的 Category 就是對(duì)裝飾模式的一種具體實(shí)現(xiàn)。它的主要作用是在不改變?cè)蓄惖那疤嵯录呃牵瑒?dòng)態(tài)地給這個(gè)類添加一些方法掏导。
好處:
a)可以減少單個(gè)文件的體積
b)可以把不同的功能組織到不同的category里
c)可以由多個(gè)開發(fā)者共同完成一個(gè)類
d)可以按需加載想要的category 等等。
category的結(jié)構(gòu)體
所有的OC類和對(duì)象羽峰,在runtime層都是用struct表示的趟咆,category也不例外添瓷,在runtime層,category用結(jié)構(gòu)體category_t.
structcategory_t {
constchar*name;//類的名字(name)
classref_t cls;//類(cls)
struct method_list_t *instanceMethods; //category中所有給類添加的實(shí)例方法的列表(instanceMethods)
struct method_list_t *classMethods;//category中所有添加的類方法的列表(classMethods)
struct protocol_list_t *protocols; //category實(shí)現(xiàn)的所有協(xié)議的列表(protocols)
struct property_list_t *instanceProperties;//category中添加的所有屬性(instanceProperties)
};
添加方法是會(huì)將方法添加進(jìn)方法列表中值纱,所以實(shí)現(xiàn)了添加方法的作用
注意鳞贷,在category中可以有屬性(property),但是該屬性只是生成了getter和setter方法的聲明虐唠,并沒(méi)有產(chǎn)生對(duì)應(yīng)的實(shí)現(xiàn)搀愧,更不會(huì)添加對(duì)應(yīng)的實(shí)例變量。如果想為實(shí)例對(duì)象添加實(shí)例變量疆偿,可以嘗試使用關(guān)聯(lián)引用技術(shù)
3咱筛、當(dāng)category方法與原類方法相同時(shí),調(diào)用方法杆故,會(huì)執(zhí)行原類方法還是category的方法迅箩?為什么
執(zhí)行category的實(shí)現(xiàn)方法
添加方法列表的時(shí)候是后添加的在新形成的列表前部,所以后編譯的在調(diào)用時(shí)會(huì)“覆蓋”前面已編譯的方法处铛。其實(shí)方法本身并沒(méi)有被覆蓋饲趋,只是調(diào)用的時(shí)候是從上而下查找方法列表,當(dāng)運(yùn)行時(shí)找到對(duì)應(yīng)的方法名后就去忙著調(diào)用了罢缸,并不會(huì)管后面的同名方法篙贸。
4、獲取view所在的viewcontroller
-(UIViewController*)viewController{
for(UIView*next =self.superview;next;next = next.superview){
UIResponder*nextResponder = [next nextResponder];
if([nextResponderisKindOfClass:[UIViewControllerclass]]){
return(UIViewController*)nextResponder;
}
}
return nil;
}
5枫疆、工廠模式和抽象工廠模式
http://www.reibang.com/p/8cf0b9804e1b
6爵川、weak底層實(shí)現(xiàn)原理
weak表其實(shí)是一個(gè)哈希表,key是所指對(duì)象的指針息楔,value是weak指針的地址數(shù)組寝贡。(value是數(shù)組的原因是:因?yàn)橐粋€(gè)對(duì)象可能被多個(gè)弱引用指針指向)
Runtime維護(hù)了一張weak表,用來(lái)存儲(chǔ)某個(gè)對(duì)象的所有的weak指針值依。
weak原理實(shí)現(xiàn)過(guò)程三步驟
1.初始化開始時(shí)圃泡,會(huì)調(diào)用objc_initWeak函數(shù),初始化新的weak指針指向?qū)ο蟮牡刂?br> 2.緊接著愿险,objc_initWeak函數(shù)里面會(huì)調(diào)用objc_storeWeak() 函數(shù)颇蜡,objc_storeWeak() 函數(shù)的作用是用來(lái)更新指針的指向,創(chuàng)建弱引用表辆亏。
3.在最后會(huì)調(diào)用clearDeallocating函數(shù)风秤。而clearDeallocating函數(shù)首先根據(jù)對(duì)象的地址獲取weak指針地址的數(shù)組,然后緊接著遍歷這個(gè)數(shù)組扮叨,將其中的數(shù)組開始置為nil缤弦,把這個(gè)entry從weak表中刪除,最后一步清理對(duì)象的記錄彻磁。
7碍沐、OC對(duì)象怎么實(shí)現(xiàn)引用計(jì)數(shù)的狸捅?
sideTable 其實(shí)是一個(gè) hash 表,下面掛了很多的 sideTable累提,sidetable 包括自旋鎖(spinlock_t)尘喝,引用計(jì)數(shù)表(refcountMap),弱引用表(weak_table_t)刻恭。
sidetables 為什么是多張表瞧省,而不是一張表扯夭?:
如果只有一張表鳍贾,如果想操作某一個(gè)兌現(xiàn)的印象計(jì)數(shù),由于不同的對(duì)象是在不同的線程操作交洗,由于不同線程需要來(lái)操作這張表骑科,所以就有資源訪問(wèn)的問(wèn)題,那么就需要對(duì)這張大表進(jìn)行加鎖操作构拳,如果成千上萬(wàn)對(duì)自己進(jìn)行引用計(jì)數(shù)操作咆爽,那么需要加鎖排隊(duì),就會(huì)有效率問(wèn)題置森,所以系統(tǒng)引用了 “分離鎖” 概念斗埂,比如 A,B同時(shí)進(jìn)行操作的話凫海,可以并發(fā)進(jìn)行呛凶,因?yàn)锳,B行贪,在不同的表中漾稀。
如果實(shí)現(xiàn)快速分流?找到當(dāng)前對(duì)象在哪張表中建瘫?:
sideTables 其實(shí)是一張 hash 表崭捍,key(對(duì)象指針)-》hash函數(shù)-》value(sideTable),通過(guò)這個(gè)hash計(jì)算之后啰脚,就可以計(jì)算出當(dāng)前對(duì)象在哪個(gè)hash表中殷蛇,也就找到了對(duì)應(yīng)的sideTable
hash 查找:
給定一個(gè)內(nèi)存地址,通過(guò)hash計(jì)算就可以得到數(shù)組的下表地址橄浓,f(ptr) = ptr%arr.count,比如內(nèi)存地址為1粒梦,通過(guò)上面的就可以找到在數(shù)組中的位置
散列表的數(shù)據(jù)結(jié)構(gòu):
自旋鎖:忙等,如果鎖已被其他線程獲取贮配,那么當(dāng)前線程會(huì)自己去不斷的獲取是否被釋放谍倦,直到其他線程釋放,適用于輕量訪問(wèn)泪勒,如+1昼蛀,-1宴猾。
引用計(jì)數(shù)表(refcountmap):其實(shí)就是hash查找,提高查找效率叼旋,插入和查找通過(guò)同一個(gè)hash函數(shù)來(lái)獲取仇哆,避免了循環(huán)遍歷。ptr->hash->size_t,其中的size_t就是引用計(jì)數(shù)值夫植,比如用64位存儲(chǔ)讹剔,第一位表示(weakly_referenced),表示對(duì)象是否存在弱引用详民,下一位表示當(dāng)前對(duì)象是都正在dealoc(deallocating)延欠,剩下的位表示引用計(jì)數(shù)值。
8沈跨、調(diào)用野指針會(huì)報(bào)錯(cuò)嗎由捎?
不一定
如果對(duì)象指針指向內(nèi)存已經(jīng)被其他對(duì)象覆寫占用,那么會(huì)crash饿凛,如果沒(méi)有沒(méi)覆寫狞玛,調(diào)用依然可以正確執(zhí)行。