IOS學(xué)習(xí)筆記--RunTime的理解
RunTime的理解
runtime:運行時刻是指一個程序在運行(或者在被執(zhí)行)的狀態(tài)滋恬。也就是說挑围,當(dāng)你打開一個程序使它在電腦上運行的時候,那個程序就是處于運行時刻铝条。在一些編程語言中靖苇,把某些可以重用的程序或者實例打包或者重建成為“運行庫"。這些實例可以在它們運行的時候被連接或者被任何程序調(diào)用班缰。
OC中runtime:是一套比較底層的純C語言API, 屬于1個C語言庫, 包含了很多底層的C語言API贤壁。 在我們平時編寫的OC代碼, 在程序運行過程時, 都會轉(zhuǎn)換成runtime的C語言代碼。比如類轉(zhuǎn)成了runtime庫里面的結(jié)構(gòu)體等數(shù)據(jù)類型埠忘,方法轉(zhuǎn)成了runtime庫里面的C語言函數(shù)脾拆,平時調(diào)方法都是轉(zhuǎn)成了objc_msgSend函數(shù)(所以說OC有個消息發(fā)送機制)
RunTime學(xué)習(xí)路線
RunTime使用前
閱讀runtime.h類
RunTime的實際使用
runtime最常用的幾個用法:【代碼塊實例以下文Button+Category為基準(zhǔn)】
動態(tài)的為一個類添加/修改/刪除 成員變量、方法
import : 成員變量莹妒、類假丧、方法
class_copyIvarList : 獲得某個類內(nèi)部的所有成員變量
unsigned int outCount = 0;
Ivar *allVars = class_copyIvarList([self class], &outCount);
for (int i = 0; i<outCount; i++) {
Ivar ivar = allVars[I];
const char * ivarName = ivar_getName(ivar);
NSLog(@"UIButton內(nèi)所有屬性變量名字 %s",ivarName);
}
class_copyMethodList : 獲得某個類內(nèi)部的所有方法
unsigned int outCount = 0;
Method *allMethods = class_copyMethodList([self class], &outCount);
for (int i = 0; i<outCount; i++) {
SEL methodSEL = method_getName(allMethods[I]);
const char * funtionName = sel_getName(methodSEL);
NSLog(@"UIButton內(nèi)所有方法名字 %s",funtionName);
}
class_getInstanceMethod : 獲得某個具體的實例方法(對象方法,減號開頭)
Method method = class_getInstanceMethod([self class], @selector(<#function_name#>));
class_getClassMethod : 獲得某個具體的類方法 (加號)
Method method = class_getClassMethod([self class], @selector(<#selector#>))
method_exchangeImplementations : 交換2個方法的實現(xiàn)
/* Method Swizzling (黑魔法)
Swizzling應(yīng)該總在+load中執(zhí)行(load類方法是程序運行時這個類被加載到內(nèi)存中就調(diào)用的一個方法动羽,執(zhí)行比較早,并且不需要我們手動調(diào)用渔期。)
Swizzling應(yīng)該總是在dispatch_once中執(zhí)行
Swizzling在+load中執(zhí)行時运吓,不要調(diào)用[super load]渴邦。如果多次調(diào)用了[super load],可能會出現(xiàn)“Swizzle無效”的假象拘哨。
為了避免Swizzling的代碼被重復(fù)執(zhí)行谋梭,我們可以通過GCD的dispatch_once函數(shù)來解決,利用dispatch_once函數(shù)內(nèi)代碼只會執(zhí)行一次的特性倦青。
*/
+(void)load {
static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{
// 交換分類的方法 layoutSubviews -> jdd_layoutSubviews
// UIButton執(zhí)行l(wèi)ayoutSubviews方法時瓮床,實際會執(zhí)行分類里的jdd_layoutSubviews , 而下文 jdd_layoutSubviews方法內(nèi)部里的[self jdd_layoutSubviews]實際會執(zhí)行UIButton原來的[self layoutSubviews] , 不會產(chǎn)生遞歸調(diào)用。
Method m1 = class_getInstanceMethod([self class], @selector(layoutSubviews));
Method m2 = class_getInstanceMethod([self class], @selector(jdd_layoutSubviews));
method_exchangeImplementations(m1, m2);
});
}