iOS底層原理——runtime

runtime

runtime 是iOS的運(yùn)行時因妙,用于實(shí)現(xiàn)iOS加載和調(diào)用屬性和方法痰憎。

函數(shù)中l(wèi)oad方法沒有使用runtime機(jī)制票髓,是底層直接調(diào)用的函數(shù)。load執(zhí)行順序是由編譯時的文件順序相同铣耘,先編譯的先執(zhí)行l(wèi)oad洽沟,類優(yōu)先于分類的順序調(diào)用 +load 方法。

initialize

+initialize 方法是在類或類的子類收到第一條消息之前被調(diào)用的蜗细,這里所指的消息包括實(shí)例方法和類方法的調(diào)用玲躯。

也就是說 +initialize 方法是以懶加載的方式被調(diào)用的,如果一直沒有給一個類或他的子類發(fā)送消息鳄乏,那么這個類的 +initialize 方法是永遠(yuǎn)不會調(diào)用的。

當(dāng)我們向某個類發(fā)送消息時棘利,runtime 會調(diào)用 IMP lookUpImpOrForward(...) 這個函數(shù)在類中查找相應(yīng)方法的實(shí)現(xiàn)或進(jìn)行消息轉(zhuǎn)發(fā)橱野,打開 objc-runtime-new.h 找到這個函數(shù):

IMP lookUpImpOrForward(Class cls, SEL sel, id inst, 
                       bool initialize, bool cache, bool resolver)
{
    Class curClass;
    IMP imp = nil;
    ...
    if (initialize  &&  !cls->isInitialized()) {
        // 類沒有初始化時,對類進(jìn)行初始化
        _class_initialize (_class_getNonMetaClass(cls, inst));
        // If sel == initialize, _class_initialize will send +initialize and 
        // then the messenger will send +initialize again after this 
        // procedure finishes. Of course, if this is not being called 
        // from the messenger then it won't happen. 2778172
    }

    ...
}

從中可以看到當(dāng)類沒有初始化時善玫,會調(diào)用 _class_initialize(Class cls) 對類進(jìn)行初始化:

void _class_initialize(Class cls)
{
    assert(!cls->isMetaClass());

    Class supercls;
    BOOL reallyInitialize = NO;

    // Make sure super is done initializing BEFORE beginning to initialize cls.
    // See note about deadlock above.
    supercls = cls->superclass;
    // 遞歸調(diào)用水援,對父類進(jìn)行_class_initialize調(diào)用,確保父類的initialize方法比子類先調(diào)用
    if (supercls  &&  !supercls->isInitialized()) {
        _class_initialize(supercls);
    }
    
    ......
    
    if (reallyInitialize) {
        // We successfully set the CLS_INITIALIZING bit. Initialize the class.
        
        // Record that we're initializing this class so we can message it.
        _setThisThreadIsInitializingClass(cls);
        
        // Send the +initialize message.
        // Note that +initialize is sent to the superclass (again) if 
        // this class doesn't implement +initialize. 2157218
        if (PrintInitializing) {
            _objc_inform("INITIALIZE: calling +[%s initialize]",
                         cls->nameForLogging());
        }
        // 發(fā)送調(diào)用類方法initialize的消息
        ((void(*)(Class, SEL))objc_msgSend)(cls, SEL_initialize);

        ......
}

在這里茅郎,先是對入?yún)⒌母割愡M(jìn)行遞歸調(diào)用蜗元,以確保父類優(yōu)先于子類初始化。

+initialize 方法在 runtime 中是以發(fā)送消息的方式調(diào)用的系冗,所以子類會覆蓋父類的實(shí)現(xiàn)奕扣,分類會覆蓋類的實(shí)現(xiàn),多個分類只會調(diào)用一個分類的 +initialize 方法掌敬。

分類

通過runtime動態(tài)將分類的屬性和方法合并到類對象惯豆,元類對象中。

  • 擴(kuò)展屬性
#import "ClassName + CategoryName.h"
#import <objc/runtime.h>

static void *strKey = &strKey;

@implementation ClassName (CategoryName) 
-(void)setStr:(NSString *)str  
{  
    objc_setAssociatedObject(self, &strKey, str, OBJC_ASSOCIATION_COPY);  
}  

-(NSString *)str  
{  
    return objc_getAssociatedObject(self, &strKey);  
}

Method Swizzling

每個類都維護(hù)一個方法(Method)列表奔害,Method則包含SEL和其對應(yīng)IMP的信息楷兽,方法交換做的事情就是把SEL和IMP的對應(yīng)關(guān)系斷開,并和新的IMP生成對應(yīng)關(guān)系华临。

交換前:Asel->AImp Bsel->BImp
交換后:Asel->BImp Bsel->AImp

方法交換之后芯杀,“方法的實(shí)現(xiàn)” 變成了 “你的處理代碼” + “方法的實(shí)現(xiàn)”

//獲取通過SEL獲取一個方法
class_getInstanceMethod
//獲取一個方法的實(shí)現(xiàn)
method_getImplementation
//獲取一個OC實(shí)現(xiàn)的編碼類型
method_getTypeEncoding
//給方法添加實(shí)現(xiàn)
class_addMethod
//用一個方法的實(shí)現(xiàn)替換另一個方法的實(shí)現(xiàn)
class_replaceMethod
//交換兩個方法的實(shí)現(xiàn)
method_exchangeImplementations
+ (void)swizzleMethods:(Class)class originalSelector:(SEL)origSel swizzledSelector:(SEL)swizSel {
    
    Method origMethod = class_getInstanceMethod(class, origSel);
    Method swizMethod = class_getInstanceMethod(class, swizSel);
    
    BOOL didAddMethod = class_addMethod(class, origSel, method_getImplementation(swizMethod), method_getTypeEncoding(swizMethod));
    if (didAddMethod) {
        class_replaceMethod(class, swizSel, method_getImplementation(origMethod), method_getTypeEncoding(origMethod));
    } else {
        method_exchangeImplementations(origMethod, swizMethod);
    }
}

調(diào)用過程,涉及到了isa指針

isa

  • arm64之前雅潭,isa是普通指針

  • arm64 isa采用共用體數(shù)據(jù)結(jié)構(gòu)揭厚,使用位域存儲更多信息,將64位數(shù)據(jù)寻馏,分開來存儲信息

  • isa共用體中的結(jié)構(gòu)體沒有實(shí)際意義,只是用來描述內(nèi)存中每一部分的作用

  • isa只有33位用來存放類地址值(shiftcls在4-36位)棋弥,需要&ISA_MASK才能像之前一樣訪問class、meta-class

  • isa共用體的好處:

    • 可以存放更多的信息
    • 更好的利用內(nèi)存空間

ps:MASK 掩碼诚欠,一般用來進(jìn)行按位(與&)運(yùn)算顽染。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末漾岳,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子粉寞,更是在濱河造成了極大的恐慌尼荆,老刑警劉巖,帶你破解...
    沈念sama閱讀 216,324評論 6 498
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件唧垦,死亡現(xiàn)場離奇詭異捅儒,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)振亮,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,356評論 3 392
  • 文/潘曉璐 我一進(jìn)店門巧还,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人坊秸,你說我怎么就攤上這事麸祷。” “怎么了褒搔?”我有些...
    開封第一講書人閱讀 162,328評論 0 353
  • 文/不壞的土叔 我叫張陵阶牍,是天一觀的道長。 經(jīng)常有香客問我星瘾,道長走孽,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,147評論 1 292
  • 正文 為了忘掉前任琳状,我火速辦了婚禮磕瓷,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘念逞。我一直安慰自己生宛,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,160評論 6 388
  • 文/花漫 我一把揭開白布肮柜。 她就那樣靜靜地躺著陷舅,像睡著了一般。 火紅的嫁衣襯著肌膚如雪审洞。 梳的紋絲不亂的頭發(fā)上莱睁,一...
    開封第一講書人閱讀 51,115評論 1 296
  • 那天,我揣著相機(jī)與錄音芒澜,去河邊找鬼仰剿。 笑死,一個胖子當(dāng)著我的面吹牛痴晦,可吹牛的內(nèi)容都是我干的南吮。 我是一名探鬼主播,決...
    沈念sama閱讀 40,025評論 3 417
  • 文/蒼蘭香墨 我猛地睜開眼誊酌,長吁一口氣:“原來是場噩夢啊……” “哼部凑!你這毒婦竟也來了露乏?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 38,867評論 0 274
  • 序言:老撾萬榮一對情侶失蹤涂邀,失蹤者是張志新(化名)和其女友劉穎瘟仿,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體比勉,經(jīng)...
    沈念sama閱讀 45,307評論 1 310
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡劳较,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,528評論 2 332
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了浩聋。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片观蜗。...
    茶點(diǎn)故事閱讀 39,688評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖衣洁,靈堂內(nèi)的尸體忽然破棺而出嫂便,到底是詐尸還是另有隱情,我是刑警寧澤闸与,帶...
    沈念sama閱讀 35,409評論 5 343
  • 正文 年R本政府宣布,位于F島的核電站岸售,受9級特大地震影響践樱,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜凸丸,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,001評論 3 325
  • 文/蒙蒙 一拷邢、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧屎慢,春花似錦瞭稼、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,657評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至集灌,卻和暖如春悔雹,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背欣喧。 一陣腳步聲響...
    開封第一講書人閱讀 32,811評論 1 268
  • 我被黑心中介騙來泰國打工腌零, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人唆阿。 一個月前我還...
    沈念sama閱讀 47,685評論 2 368
  • 正文 我出身青樓益涧,卻偏偏與公主長得像,于是被迫代替她去往敵國和親驯鳖。 傳聞我的和親對象是個殘疾皇子闲询,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,573評論 2 353