一、Runtime
1、概念:
概念:Runtime是Objective-c語言動態的核心,即運行時,在面向物件的基礎上增加了動態運行,達到很多在編譯時確定方法推遲到了運行時,從而達到動態修改、確定、交換,,,屬性及方法
作用: 這給程式員寫代碼帶來很大的靈活性,比如說你可以把訊息轉發給你想要的物件,或者隨意交換一個方法的實作之類的!多型 kvo kvc 獲得屬性方法 添加屬性方法
核心: 另外 Runtime進行訊息決議和轉發,動態呼叫程序!
只有在真正運行的時候才會根據函式的名稱找 到對應的函式來呼叫,
2、特性:撰寫的代碼具備有運行時、動態特性,從而衍生出 以下4、5
3、原理:Runtimer在Object-c的使用 程式在三個不同的層次上與運行時系統互動:
(1)通過Object-c源代碼進行互動
(2)通過NSObject類中定義的方法互動
(3)通過直接呼叫運行時函式
4、作用:
(1)在程式運行程序中,動態的創建類,動態添加、修改這個類的屬性的方法
(2)遍歷一個類中的所有成員變數、屬性、以及所有方法
(3)訊息傳遞、轉發
5、典型事例:
(1)給系統分類添加屬性、方法
(2)方法交換
(3)獲取物件的屬性、私有屬性
(4)字典轉換模型
(5)KVO、KVC
(6)(NSClassFromString class)字串
(7)block
(8)類的自我檢測
6、Objc-msgSend所做的事情
(1)找到方法的實作,由于通過單獨的類以不同方式創建相同的方法,因此這個方法的實作的確定取決于接收訊息的類的物件,也既是說多個實體類對戲那個可以創建同樣的方法,每個實體物件中該方法都是獨立存在的
(2)呼叫該方法實作,將接收訊息類指標,以及該方法的引數傳遞給這個類
(3)最后將程序的回傳值作為自己的回傳值傳遞
7、訊息傳遞的關鍵要素
(1)指向superclass指標
(2)會有一個SEL跟方法實作的
8、Msg_sender機制:先查詢本類是否又該方法的實作--->如果沒有逐級找父類,還有一個快速映射表(提高性能)---> 匹配方法 ---> 設定一個執行者---> 訊息轉發 ---> 沒有實作方法
re solveInstanceMethod 決策實力,動態方法決議
forwardingTargetForSelector 轉寄Target ,設定一個執行者 備用接收者
MethodSignatureForSelector 方法簽名,
forwardInvocation 轉寄求助,訊息重定向
doesNotRecognizeSelector 沒有找到方法 崩潰
- 先呼叫resolveInstanceMethod,如果在這里使用runtime動態添加對應的方法,并且回傳YES,訊息就找到了回應的物件,并將這個新增的方法添加到類的方法快取串列
- 如果上面的方法回傳NO的話,物件會呼叫forwardingTargetForSelector方法,以實作訊息的轉發,讓其他物件來處理這個訊息,
- 如果以上兩個方法都沒有做處理,那么物件會執行最后一個方法methodSignatureForSelector,提供一個有效的方法簽名,若提供了有效的方法簽名,程式會通過forwardInvocation方法執行簽名,若沒有提供方法簽名,觸發doesNotRecognizeSelector方法,觸發崩潰,
resolveInstanceMethod
resolveInstanceMethod是Objective-C語言中一種動態方法決議的介面,是得我們可以在運行時動態的為一個selector提供實作,我們只需要實作 +resolveInstanceMethod和+resolveClassMethod方法,并在其中為指定的selector提供實作即可(通過呼叫運行時函式class_addMethod來添加),這兩個方法都是NSObject中的類方法,其原型為:
+ (BOOL)resolveClassMethod:(SEL)name;
+ (BOOL)resolveInstanceMethod:(SEL)name;
引數那么是需要被動態決議的selector;如果在該函式中為指定的selector提供實作,無論回傳YES還是NO,編譯運行都是正確的,如果在該函式內并沒有真正的為selector提供實作,如果回傳YES,運行會crash,其原理很簡單,因為當前類既沒有為selector提供實作,又沒有實作訊息轉發,自然會crash,
forwardingTargetForSelector
forwardingTargetForSelector是Objective-C語言中訊息快速重定向的函式,開發者可以在派生類中對其進行多載,從而將無法處理的selector轉發給另一個物件,
methodSignatureForSelector
methodSigntureForSelector的作用在在于為另一個類實作的訊息創建一個有效的方法簽名,如果沒有實作有效的方法簽名,程式就會崩潰
forwardInvocation
在回傳有效的方法簽名的情況下,當前物件則會呼叫forwardInvocation方法,以完成訊息的最終傳遞,
1、動態決議的一個例子
2、備用接受者
3.重簽名
二、運行時常用的API:
objc_*
objc_系列函式關注于宏觀使用,如類與協議的空間分配,注冊,注銷等操作
// 1.objc_xxx 系列函式
// 函式名稱 函式作用
objc_getClass 獲取Class物件
objc_getMetaClass 獲取MetaClass物件
objc_allocateClassPair 分配空間,創建類(僅在 創建之后,注冊之前 能夠添加成員變數)
objc_registerClassPair 注冊一個類(注冊后方可使用該類創建物件)
objc_disposeClassPair 注銷某個類
objc_allocateProtocol 開辟空間創建協議
objc_registerProtocol 注冊一個協議
objc_constructInstance 構造一個實體物件(ARC下無效)
objc_destructInstance 析構一個實體物件(ARC下無效)
objc_setAssociatedObject 為實體物件關聯物件
objc_getAssociatedObje*ct 獲取實體物件的關聯物件
objc_removeAssociatedObjects 清空實體物件的所有關聯物件
class_*
class_系列函式關注于類的內部,如實體變數,屬性,方法,協議等相關問題
// 2.class_xxx 系列函式
函式名稱 函式作用
class_addIvar 為類添加實體變數
class_addProperty 為類添加屬性
class_addMethod 為類添加方法
class_addProtocol 為類遵循協議
class_replaceMethod 替換類某方法的實作
class_getName 獲取類名
class_isMetaClass 判斷是否為元類
objc_getProtocol 獲取某個協議
objc_copyProtocolList 拷貝在運行時中注冊過的協議串列
class_getSuperclass 獲取某類的父類
class_setSuperclass 設定某類的父類
class_getProperty 獲取某類的屬性
class_getInstanceVariable 獲取實體變數
class_getClassVariable 獲取類變數
class_getInstanceMethod 獲取實體方法
class_getClassMethod 獲取類方法
class_getMethodImplementation 獲取方法的實作
class_getInstanceSize 獲取類的實體的大小
class_respondsToSelector 判斷類是否實作某方法
class_conformsToProtocol 判斷類是否遵循某協議
class_createInstance 創建類的實體
class_copyIvarList 拷貝類的實體變數串列
class_copyMethodList 拷貝類的方法串列
class_copyProtocolList 拷貝類遵循的協議串列
class_copyPropertyList 拷貝類的屬性串列
objcet_*
objcet_系列函式關注于物件的角度,如實體變數
// 3.object_xxx 系列函式
函式名稱 函式作用
object_copy 物件copy(ARC無效)
object_dispose 物件釋放(ARC無效)
object_getClassName 獲取物件的類名
object_getClass 獲取物件的Class
object_setClass 設定物件的Class
object_getIvar 獲取物件中實體變數的值
object_setIvar 設定物件中實體變數的值
object_getInstanceVariable 獲取物件中實體變數的值 (ARC中無效,使用object_getIvar)
object_setInstanceVariable 設定物件中實體變數的值 (ARC中無效,使用object_setIvar)
method_*
method_系列函式關注于方法內部,如果方法的引數及回傳值型別和方法的實作
// 4.method_xxx 系列函式
函式名稱 函式作用
method_getName 獲取方法名
method_getImplementation 獲取方法的實作
method_getTypeEncoding 獲取方法的型別編碼
method_getNumberOfArguments 獲取方法的引數個數
method_copyReturnType 拷貝方法的回傳型別
method_getReturnType 獲取方法的回傳型別
method_copyArgumentType 拷貝方法的引數型別
method_getArgumentType 獲取方法的引數型別
method_getDescription 獲取方法的描述
method_setImplementation 設定方法的實作
method_exchangeImplementations 替換方法的實作
property_*
property_系類函式關注與屬性*內部,如屬性的特性等
// 5.property_xxx 系列函式
函式名稱 函式作用
property_getName 獲取屬性名
property_getAttributes 獲取屬性的特性串列
property_copyAttributeList 拷貝屬性的特性串列
property_copyAttributeValue 拷貝屬性中某特性的值
protocol_*
// 6.protocol_xxx 系列函式
函式名稱 函式作用
protocol_conformsToProtocol 判斷一個協議是否遵循另一個協議
protocol_isEqual 判斷兩個協議是否一致
protocol_getName 獲取協議名稱
protocol_copyPropertyList 拷貝協議的屬性串列
protocol_copyProtocolList 拷貝某協議所遵循的協議串列
protocol_copyMethodDescriptionList 拷貝協議的方法串列
protocol_addProtocol 為一個協議遵循另一協議
protocol_addProperty 為協議添加屬性
protocol_getProperty 獲取協議中的某個屬性
protocol_addMethodDescription 為協議添加方法描述
protocol_getMethodDescription 獲取協議中某方法的描述
ivar_*
// 7.ivar_xxx 系列函式
函式名稱 函式作用
ivar_getName 獲取Ivar名稱
ivar_getTypeEncoding 獲取型別編碼
ivar_getOffset 獲取偏移量
sel_*
// 8.sel_xxx 系列函式
函式名稱 函式作用
sel_getName 獲取名稱
sel_getUid 注冊方法
sel_registerName 注冊方法
sel_isEqual 判斷方法是否相等
imp_*
// 9.imp_xxx 系列函式
函式名稱 函式作用
imp_implementationWithBlock 通過代碼塊創建IMP
imp_getBlock 獲取函式指標中的代碼塊
imp_removeBlock 移除IMP中的代碼塊
轉載請註明出處,本文鏈接:https://www.uj5u.com/yidong/503023.html
標籤:iOS
