溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

如何使用objcruntime實現iOS閉環的懶加載功能

發布時間:2021-09-28 10:46:44 來源:億速云 閱讀:149 作者:小新 欄目:編程語言

這篇文章主要為大家展示了“如何使用objcruntime實現iOS閉環的懶加載功能”,內容簡而易懂,條理清晰,希望能夠幫助大家解決疑惑,下面讓小編帶領大家一起研究并學習一下“如何使用objcruntime實現iOS閉環的懶加載功能”這篇文章吧。

使用objc runtime實現懶加載

地址:AutoPropertyCocoa

懶加載形式如下

- (id)lazyloadProperty{  if(_lazyloadProperty == nil){    _lazyloadProperty = [XClass ...];  }  return _lazyloadProperty;}

一般使用宏定義可以輕松完成。但是沒有一致性,移植差。

利用objc runtime的動態性實現懶加載可以實現即可增加又可刪除功能,也可以避免污染類型。該三方彌補了目前沒有閉環實現懶加載三方的空缺。

主要流程:

  1. 實例或者類的懶加載如果是實例對象則鉤住并修改類型將其子類化對該類型進行method swizzling如果現在進行解綁,則判斷是否是自己實現的方法.如果是自己實現的方法->5,否則->6調用method swizzling還原刪除這個類型的這個方法

難點:

自己實現method swizzling

  1. 重新實現objc1時代的方法class_removeMethods鉤住運行時中的runtimelock,實現修改類型數據時的安全性

我們再實現method swizzling時的兩個API

OBJC_EXPORT IMP _Nullableclass_replaceMethod(Class _Nullable cls, SEL _Nonnull name, IMP _Nonnull imp,          const char * _Nullable types)  OBJC_AVAILABLE(10.5, 2.0, 9.0, 1.0, 2.0);OBJC_EXPORT voidmethod_exchangeImplementations(Method _Nonnull m1, Method _Nonnull m2)  OBJC_AVAILABLE(10.5, 2.0, 9.0, 1.0, 2.0);

不管使用哪種,如果這個類型沒有實現該方法而是父類實現的話,就需要動態增加一個方法。動態增加的方法在Objc1時代,是可以通過下列方法刪除的:

OBJC_EXPORT voidclass_removeMethods(Class _Nullable, struct objc_method_list * _Nonnull)  OBJC2_UNAVAILABLE;

Objc2時代之后runtime被重寫后沒有該方法了,并且新的runtime的類結構看起來就沒打算讓開發者刪除方法,所以這里將過程記下。

首先看類讀寫器的結構class_rw_t

struct class_rw_t {  // Be warned that Symbolication knows the layout of this structure.  uint32_t flags;  uint32_t version;  const class_ro_t *ro;  method_array_t methods;//刪除這里的一個方法  property_array_t properties;  protocol_array_t protocols;  Class firstSubclass;  Class nextSiblingClass;  char *demangledName;#if SUPPORT_INDEXED_ISA  uint32_t index;#endif};

method_array_t繼承于list_array_tt<method_t, method_list_t>,它是數組結構。存儲的內容是method_list_t.

method_list_t又繼承于entsize_list_tt<method_t, method_list_t, 0x3>,他也是數組結構。

整個method_array_t結構是二維數組。每次刪掉一個method_t需要用新method_list_t替換原對象。

然后是線程安全的問題,需要獲取到蘋果在操作類型的時候使用的讀寫鎖(pthread_rw_lock_t runtimelock)。沒有這把鎖任何對runtime的修改都是不可靠的。

最終采取的方式是:劫持暴露了符號的系統函數然后阻塞線程

劫持系統C函數使用的是臉書的魚鉤,這個鉤子在macOS其實也是可以正常工作的。

剩下的就是尋找合適的函數了,這函數要滿足兩個條件:

  1. 該函數在符號表中存在函數內部在lock runtimelock之后存在滿足條件1的第二個函數

找了半天發現最合適的只有objc_allocateProtocol()了,objc_allocateProtocol內部會調用calloc(),所以第二個被劫持函數就是calloc。為了減小calloc的開銷,需要稍微做一些工作。

  1. 對每次調用進行比較線程ID的操作顯然比暴力阻塞線程好。減小劫持后的calloc的調用棧。

以上是“如何使用objcruntime實現iOS閉環的懶加載功能”這篇文章的所有內容,感謝各位的閱讀!相信大家都有了一定的了解,希望分享的內容對大家有所幫助,如果還想學習更多知識,歡迎關注億速云行業資訊頻道!

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

亚洲午夜精品一区二区_中文无码日韩欧免_久久香蕉精品视频_欧美主播一区二区三区美女