_objc_msgForward
是 IMP 類型,用於消息轉發的:當向一個對象發送一條消息,但它並無實現的時候,_objc_msgForward
會嘗試作消息轉發。html
咱們能夠這樣建立一個_objc_msgForward
對象:ios
IMP msgForwardIMP = _objc_msgForward;
git
在上篇中的《objc中向一個對象發送消息[obj foo]
和objc_msgSend()
函數之間有什麼關係?》曾提到objc_msgSend
在「消息傳遞」中的做用。在「消息傳遞」過程當中,objc_msgSend
的動做比較清晰:首先在 Class 中的緩存查找 IMP (沒緩存則初始化緩存),若是沒找到,則向父類的 Class 查找。若是一直查找到根類仍舊沒有實現,則用_objc_msgForward
函數指針代替 IMP 。最後,執行這個 IMP 。github
Objective-C運行時是開源的,因此咱們能夠看到它的實現。打開 Apple Open Source 裏Mac代碼裏的obj包 下載一個最新版本,找到objc-runtime-new.mm
,進入以後搜索_objc_msgForward
。面試
_objc_msgForward
的功能解釋:
/***********************************************************************
* lookUpImpOrForward.
* The standard IMP lookup.
* initialize==NO tries to avoid +initialize (but sometimes fails)
* cache==NO skips optimistic unlocked lookup (but uses cache elsewhere)
* Most callers should use initialize==YES and cache==YES.
* inst is an instance of cls or a subclass thereof, or nil if none is known.
* If cls is an un-initialized metaclass then a non-nil inst is faster.
* May return _objc_msgForward_impcache. IMPs destined for external use
* must be converted to _objc_msgForward or _objc_msgForward_stret.
* If you don't want forwarding at all, use lookUpImpOrNil() instead. **********************************************************************/ 複製代碼
對 objc-runtime-new.mm
文件裏與_objc_msgForward
有關的三個函數使用僞代碼展現下:編程
// objc-runtime-new.mm 文件裏與 _objc_msgForward 有關的三個函數使用僞代碼展現
// Created by https://github.com/ChenYilong
// Copyright (c) 微博@iOS程序犭袁(http://weibo.com/luohanchenyilong/). All rights reserved.
// 同時,這也是 obj_msgSend 的實現過程
id objc_msgSend(id self, SEL op, ...) {
if (!self) return nil;
IMP imp = class_getMethodImplementation(self->isa, SEL op);
imp(self, op, ...); //調用這個函數,僞代碼...
}
//查找IMP
IMP class_getMethodImplementation(Class cls, SEL sel) {
if (!cls || !sel) return nil;
IMP imp = lookUpImpOrNil(cls, sel);
if (!imp) return _objc_msgForward; //_objc_msgForward 用於消息轉發
return imp;
}
IMP lookUpImpOrNil(Class cls, SEL sel) {
if (!cls->initialize()) {
_class_initialize(cls);
}
Class curClass = cls;
IMP imp = nil;
do { //先查緩存,緩存沒有時重建,仍舊沒有則向父類查詢
if (!curClass) break;
if (!curClass->cache) fill_cache(cls, curClass);
imp = cache_getImp(curClass, sel);
if (imp) break;
} while (curClass = curClass->superclass);
return imp;
}
複製代碼
雖然Apple沒有公·開_objc_msgForward·的實現源碼,可是咱們仍是能得出結論:api
_objc_msgForward
是一個函數指針(和 IMP 的類型同樣),是用於消息轉發的:當向一個對象發送一條消息,但它並無實現的時候,_objc_msgForward
會嘗試作消息轉發。數組
在上篇中的《objc中向一個對象發送消息
[obj foo]
和objc_msgSend()
函數之間有什麼關係?》曾提到objc_msgSend
在「消息傳遞」中的做用。在「消息傳遞」過程當中,objc_msgSend
的動做比較清晰:首先在 Class 中的緩存查找 IMP (沒緩存則初始化緩存),若是沒找到,則向父類的 Class 查找。若是一直查找到根類仍舊沒有實現,則用_objc_msgForward
函數指針代替 IMP 。最後,執行這個 IMP 。緩存
爲了展現消息轉發的具體動做,這裏嘗試向一個對象發送一條錯誤的消息,並查看一下_objc_msgForward
是如何進行轉發的。安全
首先開啓調試模式、打印出全部運行時發送的消息: 能夠在代碼裏執行下面的方法:
(void)instrumentObjcMessageSends(YES);
由於該函數處於 objc-internal.h內,而該文件並不開放,因此調用的時候先聲明,目的是告訴編譯器程序目標文件包含該方法存在,讓編譯經過
OBJC_EXPORT void
instrumentObjcMessageSends(BOOL flag)
OBJC_AVAILABLE(10.0, 2.0, 9.0, 1.0, 2.0);
複製代碼
或者斷點暫停程序運行,並在 gdb 中輸入下面的命令:
call (void)instrumentObjcMessageSends(YES)
以第二種爲例,操做以下所示:
以後,運行時發送的全部消息都會打印到/tmp/msgSend-xxxx
文件裏了。 終端中輸入命令前往:
open /private/tmp
可能看到有多條,找到最新生成的,雙擊打開
在模擬器上執行執行如下語句(這一套調試方案僅適用於模擬器,真機不可用,關於該調試方案的拓展連接: Can the messages sent to an object in Objective-C be monitored or printed out? ),向一個對象發送一條錯誤的消息:
//
// main.m
// CYLObjcMsgForwardTest
//
// Created by http://weibo.com/luohanchenyilong/.
// Copyright (c) 2015年 微博@iOS程序犭袁. All rights reserved.
//
#import <UIKit/UIKit.h>
#import "AppDelegate.h"
#import "CYLTest.h"
int main(int argc, char * argv[]) {
@autoreleasepool {
CYLTest *test = [[CYLTest alloc] init];
[test performSelector:(@selector(iOS程序犭袁))];
return UIApplicationMain(argc, argv, nil, NSStringFromClass([AppDelegate class]));
}
}
複製代碼
你能夠在
/tmp/msgSend-xxxx
(我這一次是/tmp/msgSend-9805)文件裏,看到打印出來:
+ CYLTest NSObject initialize
+ CYLTest NSObject alloc
- CYLTest NSObject init
- CYLTest NSObject performSelector:
+ CYLTest NSObject resolveInstanceMethod:
+ CYLTest NSObject resolveInstanceMethod:
- CYLTest NSObject forwardingTargetForSelector:
- CYLTest NSObject forwardingTargetForSelector:
- CYLTest NSObject methodSignatureForSelector:
- CYLTest NSObject methodSignatureForSelector:
- CYLTest NSObject class
- CYLTest NSObject doesNotRecognizeSelector:
- CYLTest NSObject doesNotRecognizeSelector:
- CYLTest NSObject class
複製代碼
結合《NSObject官方文檔》,排除掉 NSObject 作的事,剩下的就是_objc_msgForward
消息轉發作的幾件事:
resolveInstanceMethod:
方法 (或 resolveClassMethod:
)。容許用戶在此時爲該 Class 動態添加實現。若是有實現了,則調用並返回YES,那麼從新開始objc_msgSend
流程。這一次對象會響應這個選擇器,通常是由於它已經調用過class_addMethod
。若是仍沒實現,繼續下面的動做。forwardingTargetForSelector:
方法,嘗試找到一個能響應該消息的對象。若是獲取到,則直接把消息轉發給它,返回非 nil 對象。不然返回 nil ,繼續下面的動做。注意,這裏不要返回 self ,不然會造成死循環。methodSignatureForSelector:
方法,嘗試得到一個方法簽名。若是獲取不到,則直接調用doesNotRecognizeSelector
拋出異常。若是能獲取,則返回非nil:建立一個 NSlnvocation 並傳給forwardInvocation:
。forwardInvocation:
方法,將第3步獲取到的方法簽名包裝成 Invocation 傳入,如何處理就在這裏面了,並返回非nil。doesNotRecognizeSelector:
,默認的實現是拋出異常。若是第3步沒能得到一個方法簽名,執行該步驟。上面前4個方法均是模板方法,開發者能夠override
,由 runtime
來調用。最多見的實現消息轉發:就是重寫方法3和4,吞掉一個消息或者代理給其餘對象都是沒問題的
也就是說_objc_msgForward
在進行消息轉發的過程當中會涉及如下這幾個方法:
resolveInstanceMethod:
方法 (或 resolveClassMethod:
)。forwardingTargetForSelector:
方法methodSignatureForSelector:
方法orwardInvocation:
方法doesNotRecognizeSelector:
方法爲了能更清晰地理解這些方法的做用,git倉庫裏也給出了一個Demo,名稱叫「 _objc_msgForward_demo
」,可運行起來看看。
下面回答下第二個問題「直接_objc_msgForward
調用它將會發生什麼?」
直接調用_objc_msgForward
是很是危險的事,若是用很差會直接致使程序Crash,可是若是用得好,能作不少很是酷的事。
就好像跑酷,幹得好,叫「耍酷」,幹很差就叫「做死」。
正如前文所說:
_objc_msgForward
是 IMP 類型,用於消息轉發的:當向一個對象發送一條消息,但它並無實現的時候,_objc_msgForward
會嘗試作消息轉發。
如何調用_objc_msgForward
? _objc_msgForward
隸屬 C 語言,有三個參數 :
-- | _objc_msgForward 參數 |
類型 |
---|---|---|
1. | 所屬對象 | id類型 |
2. | 方法名 | SEL類型 |
3. | 可變參數 | 可變參數類型 |
首先了解下如何調用 IMP 類型的方法,IMP類型是以下格式:
爲了直觀,咱們能夠經過以下方式定義一個 IMP類型 :
typedef void (*voidIMP)(id, SEL, ...)
一旦調用_objc_msgForward
,將跳過查找 IMP 的過程,直接觸發「消息轉發」,
若是調用了_objc_msgForward
,即便這個對象確實已經實現了這個方法,你也會告訴objc_msgSend:
「我沒有在這個對象裏找到這個方法的實現」
有哪些場景須要直接調用_objc_msgForward
?最多見的場景是:你想獲取某方法所對應的NSInvocation對象。舉例說明:
JSPatch (Github連接)就是直接調用_objc_msgForward
來實現其核心功能的:
JSPatch 以小巧的體積作到了讓JS調用/替換任意OC方法,讓iOS APP具有熱更新的能力。
做者的博文《JSPatch實現原理詳解》詳細記錄了實現原理,有興趣能夠看下。
同時 RAC(ReactiveCocoa) 源碼中也用到了該方法。
runtime 對註冊的類, 會進行佈局,對於 weak 對象會放入一個 hash 表中。 用 weak 指向的對象內存地址做爲 key,當此對象的引用計數爲0的時候會 dealloc,假如 weak 指向的對象內存地址是a,那麼就會以a爲鍵, 在這個 weak 表中搜索,找到全部以a爲鍵的 weak 對象,從而設置爲 nil。 在上篇中的《runtime 如何實現 weak 屬性》有論述。(注:在上篇的《使用runtime Associate方法關聯的對象,須要在主對象dealloc的時候釋放麼?》裏給出的「對象的內存銷燬時間表」也提到__weak引用的解除時間。)
咱們能夠設計一個函數(僞代碼)來表示上述機制:
objc_storeWeak(&a, b)
函數:
objc_storeWeak
函數把第二個參數--賦值對象(b)的內存地址做爲鍵值key,將第一個參數--weak修飾的屬性變量(a)的內存地址(&a)做爲value,註冊到 weak 表中。若是第二個參數(b)爲0(nil),那麼把變量(a)的內存地址(&a)從weak表中刪除,
你能夠把objc_storeWeak(&a, b)
理解爲:objc_storeWeak(value, key)
,而且當key變nil,將value置nil。
在b非nil時,a和b指向同一個內存地址,在b變nil時,a變nil。此時向a發送消息不會崩潰:在Objective-C中向nil發送消息是安全的。
而若是a是由assign修飾的,則: 在b非nil時,a和b指向同一個內存地址,在b變nil時,a仍是指向該內存地址,變野指針。此時向a發送消息極易崩潰。
下面咱們將基於objc_storeWeak(&a, b)
函數,使用僞代碼模擬「runtime如何實現weak屬性」:
// 使用僞代碼模擬:runtime如何實現weak屬性
// http://weibo.com/luohanchenyilong/
// https://github.com/ChenYilong
id obj1;
objc_initWeak(&obj1, obj);
/*obj引用計數變爲0,變量做用域結束*/
objc_destroyWeak(&obj1);
複製代碼
下面對用到的兩個方法objc_initWeak和objc_destroyWeak
作下解釋:
整體說來,做用是: 經過objc_initWeak
函數初始化「附有weak修飾符的變量(obj1)」,在變量做用域結束時經過objc_destoryWeak
函數釋放該變量(obj1)。
下面分別介紹下方法的內部實現:
objc_initWeak
函數的實現是這樣的:在將「附有weak修飾符的變量(obj1)」初始化爲0(nil)後,會將「賦值對象」(obj)做爲參數,調用objc_storeWeak
函數。
obj1 = 0;
obj_storeWeak(&obj1, obj);
複製代碼
也就是說:
weak 修飾的指針默認值是 nil (在Objective-C中向nil發送消息是安全的)
而後obj_destroyWeak
函數將0(nil)做爲參數,調用objc_storeWeak
函數。
objc_storeWeak(&obj1, 0)
;
前面的源代碼與下列源代碼相同。
// 使用僞代碼模擬:runtime如何實現weak屬性
// http://weibo.com/luohanchenyilong/
// https://github.com/ChenYilong
id obj1;
obj1 = 0;
objc_storeWeak(&obj1, obj);
/* ... obj的引用計數變爲0,被置nil ... */
objc_storeWeak(&obj1, 0);
複製代碼
objc_storeWeak
函數把第二個參數--賦值對象(obj)的內存地址做爲鍵值,將第一個參數--weak修飾的屬性變量(obj1)的內存地址註冊到 weak 表中。若是第二個參數(obj)爲0(nil),那麼把變量(obj1)的地址從weak表中刪除。
解釋下:
objc_ivar_list
實例變量的鏈表 和 instance_size
實例變量的內存大小已經肯定,同時runtime 會調用 class_setIvarLayout
或 class_setWeakIvarLayout
來處理 strong weak 引用。因此不能向存在的類中添加實例變量;class_addIvar
函數。可是得在調用 objc_allocateClassPair
以後,objc_registerClassPair
以前,緣由同上。總的說來,Run loop,正如其名,loop表示某種循環,和run放在一塊兒就表示一直在運行着的循環。實際上,run loop和線程是緊密相連的,能夠這樣說run loop是爲了線程而生,沒有線程,它就沒有存在的必要。Run loops是線程的基礎架構部分, Cocoa 和 CoreFundation 都提供了 run loop 對象方便配置和管理線程的 run loop (如下都以 Cocoa 爲例)。每一個線程,包括程序的主線程( main thread )都有與之相應的 run loop 對象。
runloop 和線程的關係:
iOS的應用程序裏面,程序啓動後會有一個以下的main()函數
int main(int argc, char * argv[]) {
@autoreleasepool {
return UIApplicationMain(argc, argv, nil, NSStringFromClass([AppDelegate class]));
}
}
複製代碼
重點是UIApplicationMain()函數,這個方法會爲main thread設置一個NSRunLoop對象,這就解釋了:爲何咱們的應用能夠在無人操做的時候休息,須要讓它幹活的時候又能立馬響應。
NSRunLoop *runloop = [NSRunLoop currentRunLoop];
參考連接:《Objective-C之run loop詳解》。
model 主要是用來指定事件在運行循環中的優先級的,分爲:
蘋果公開提供的 Mode 有兩個:
RunLoop只能運行在一種mode下,若是要換mode,當前的loop也須要停下重啓成新的。利用這個機制,ScrollView滾動過程當中NSDefaultRunLoopMode(kCFRunLoopDefaultMode)的mode會切換到UITrackingRunLoopMode來保證ScrollView的流暢滑動:只能在NSDefaultRunLoopMode模式下處理的事件會影響ScrollView的滑動。
若是咱們把一個NSTimer對象以NSDefaultRunLoopMode(kCFRunLoopDefaultMode)添加到主運行循環中的時候, ScrollView滾動過程當中會由於mode的切換,而致使NSTimer將再也不被調度。
同時由於mode仍是可定製的,因此:
Timer計時會被scrollView的滑動影響的問題能夠經過將timer添加到NSRunLoopCommonModes(kCFRunLoopCommonModes)來解決。代碼以下:
//
// http://weibo.com/luohanchenyilong/ (微博@iOS程序犭袁)
// https://github.com/ChenYilong
//將timer添加到NSDefaultRunLoopMode中
[NSTimer scheduledTimerWithTimeInterval:1.0
target:self
selector:@selector(timerTick:)
userInfo:nil
repeats:YES];
//而後再添加到NSRunLoopCommonModes裏
NSTimer *timer = [NSTimer timerWithTimeInterval:1.0
target:self
selector:@selector(timerTick:)
userInfo:nil
repeats:YES];
[[NSRunLoop currentRunLoop] addTimer:timer forMode:NSRunLoopCommonModes];
複製代碼
通常來說,一個線程一次只能執行一個任務,執行完成後線程就會退出。若是咱們須要一個機制,讓線程能隨時處理事件但並不退出,一般的代碼邏輯 是這樣的:
function loop() {
initialize();
do {
var message = get_next_message();
process_message(message);
} while (message != quit);
}
複製代碼
或使用僞代碼來展現下:
//
// http://weibo.com/luohanchenyilong/ (微博@iOS程序犭袁)
// https://github.com/ChenYilong
int main(int argc, char * argv[]) {
//程序一直運行狀態
while (AppIsRunning) {
//睡眠狀態,等待喚醒事件
id whoWakesMe = SleepForWakingUp();
//獲得喚醒事件
id event = GetEvent(whoWakesMe);
//開始處理事件
HandleEvent(event);
}
return 0;
}
複製代碼
參考連接:
經過 retainCount 的機制來決定對象是否須要釋放。 每次 runloop 的時候,都會檢查對象的 retainCount,若是retainCount 爲 0,說明該對象沒有地方須要繼續使用了,能夠釋放掉了。
ARC相對於MRC,不是在編譯時添加retain/release/autorelease這麼簡單。應該是編譯期和運行期兩部分共同幫助開發者管理內存。
在編譯期,ARC用的是更底層的C接口實現的retain/release/autorelease,這樣作性能更好,也是爲何不能在ARC環境下手動retain/release/autorelease,同時對同一上下文的同一對象的成對retain/release操做進行優化(即忽略掉沒必要要的操做);ARC也包含運行期組件,這個地方作的優化比較複雜,但也不能被忽略。
分兩種狀況:手動干預釋放時機、系統自動去釋放。
釋放的時機總結起來,能夠用下圖來表示:
下面對這張圖進行詳細的解釋:從程序啓動到加載完成是一個完整的運行循環,而後會停下來,等待用戶交互,用戶的每一次交互都會啓動一次運行循環,來處理用戶全部的點擊事件、觸摸事件。
咱們都知道: 全部 autorelease 的對象,在出了做用域以後,會被自動添加到最近建立的自動釋放池中。
可是若是每次都放進應用程序的 main.m 中的 autoreleasepool 中,早晚有被撐滿的一刻。這個過程當中一定有一個釋放的動做。什麼時候?
在一次完整的運行循環結束以前,會被銷燬。
那什麼時間會建立自動釋放池?運行循環檢測到事件並啓動後,就會建立自動釋放池。
從 RunLoop 源代碼中可知,子線程默認是沒有 RunLoop 的,若是須要在子線程開啓 RunLoop ,則須要調用 [NSRunLoop CurrentRunLoop] 方法,它內部實現是先檢查線程,若是發現是子線程,以懶加載的形式 建立一個子線程的 RunLoop。並存儲在一個全局的 可變字典裏。編程人員在調用 [NSRunLoop CurrentRunLoop] 時,是自動建立 RunLoop 的,而無法手動建立。
自定義的 NSOperation 和 NSThread 須要手動建立自動釋放池。好比: 自定義的 NSOperation 類中的 main 方法裏就必須添加自動釋放池。不然出了做用域後,自動釋放對象會由於沒有自動釋放池去處理它,而形成內存泄露。
但對於 blockOperation 和 invocationOperation 這種默認的Operation ,系統已經幫咱們封裝好了,不須要手動建立自動釋放池。
@autoreleasepool 當自動釋放池被銷燬或者耗盡時,會向自動釋放池中的全部對象發送 release 消息,釋放自動釋放池中的全部對象。
若是在一個vc的viewDidLoad中建立一個 Autorelease對象,那麼該對象會在 viewDidAppear 方法執行前就被銷燬了。
參考連接:《黑幕背後的Autorelease》
訪問了懸垂指針,好比對一個已經釋放的對象執行了release、訪問已經釋放對象的成員變量或者發消息。 死循環
autoreleasepool 以一個隊列數組的形式實現,主要經過下列三個函數完成.
objc_autoreleasepoolPush
objc_autoreleasepoolPop
objc_autorelease
看函數名就能夠知道,對 autorelease 分別執行 push,和 pop 操做。銷燬對象時執行release操做。
舉例說明:咱們都知道用類方法建立的對象都是 Autorelease 的,那麼一旦 Person 出了做用域,當在 Person 的 dealloc 方法中打上斷點,咱們就能夠看到這樣的調用堆棧信息:
一個對象中強引用了block,在block中又強引用了該對象,就會發射循環引用。
解決方法是將該對象使用__weak或者__block修飾符修飾以後再在block中使用。
檢測代碼中是否存在循環引用問題,可以使用 Facebook 開源的一個檢測工具 FBRetainCycleDetector 。
默認狀況下,在block中訪問的外部變量是複製過去的,即:寫操做不對原變量生效。可是你能夠加上 __block 來讓其寫操做生效,示例代碼以下:
__block int a = 0;
void (^foo)(void) = ^{
a = 1;
};
foo();
//這裏,a的值被修改成1
複製代碼
這是 微博@唐巧_boy的《iOS開發進階》中的第11.2.3章節中的描述。你一樣能夠在面試中這樣回答,但你並無答到「點子上」。真正的緣由,並無書這本書裏寫的這麼「神奇」,並且這種說法也有點牽強。面試官確定會追問「爲何寫操做就生效了?」真正的緣由是這樣的:
咱們都知道:Block不容許修改外部變量的值,這裏所說的外部變量的值,指的是棧中指針的內存地址。__block 所起到的做用就是隻要觀察到該變量被 block 所持有,就將「外部變量」在棧中的內存地址放到了堆中。進而在block內部也能夠修改外部變量的值。
Block不容許修改外部變量的值。Apple這樣設計,應該是考慮到了block的特殊性,block也屬於「函數」的範疇,變量進入block,實際就是已經改變了做用域。在幾個做用域之間進行切換時,若是不加上這樣的限制,變量的可維護性將大大下降。又好比我想在block內聲明瞭一個與外部同名的變量,此時是容許呢仍是不容許呢?只有加上了這樣的限制,這樣的情景才能實現。因而棧區變成了紅燈區,堆區變成了綠燈區。
咱們能夠打印下內存地址來進行驗證:
__block int a = 0;
NSLog(@"定義前:%p", &a); //棧區
void (^foo)(void) = ^{
a = 1;
NSLog(@"block內部:%p", &a); //堆區
};
NSLog(@"定義後:%p", &a); //堆區
foo();
複製代碼
2016-05-17 02:03:33.559 LeanCloudChatKit-iOS[1505:713679] 定義前:0x16fda86f8
2016-05-17 02:03:33.559 LeanCloudChatKit-iOS[1505:713679] 定義後:0x155b22fc8
2016-05-17 02:03:33.559 LeanCloudChatKit-iOS[1505:713679] block內部: 0x155b22fc8
複製代碼
「定義後」和「block內部」二者的內存地址是同樣的,咱們都知道 block 內部的變量會被 copy 到堆區,「block內部」打印的是堆地址,於是也就能夠知道,「定義後」打印的也是堆的地址。
那麼如何證實「block內部」打印的是堆地址?
把三個16進制的內存地址轉成10進制就是:
中間相差438851376個字節,也就是 418.5M 的空間,由於堆地址要小於棧地址,又由於iOS中一個進程的棧區內存只有1M,Mac也只有8M,顯然a已是在堆區了。
這也證明了:a 在定義前是棧區,但只要進入了 block 區域,就變成了堆區。這纔是 __block 關鍵字的真正做用。
__block 關鍵字修飾後,int類型也從4字節變成了32字節,這是 Foundation 框架 malloc 出來的。這也一樣能證明上面的結論。(PS:竟然比 NSObject alloc 出來的 16 字節要多一倍)。
理解到這是由於堆棧地址的變動,而非所謂的「寫操做生效」,這一點相當重要,要否則你如何解釋下面這個現象:
如下代碼編譯能夠經過,而且在block中成功將a的從Tom修改成Jerry。
NSMutableString *a = [NSMutableString stringWithString:@"Tom"];
NSLog(@"\n 定之前:------------------------------------\n\ a指向的堆中地址:%p;a在棧中的指針地址:%p", a, &a); //a在棧區
void (^foo)(void) = ^{
a.string = @"Jerry";
NSLog(@"\n block內部:------------------------------------\n\ a指向的堆中地址:%p;a在棧中的指針地址:%p", a, &a); //a在棧區
//a = [NSMutableString stringWithString:@"William"];
};
foo();
NSLog(@"\n 定之後:------------------------------------\n\ a指向的堆中地址:%p;a在棧中的指針地址:%p", a, &a); //a在棧區
![](https://user-gold-cdn.xitu.io/2019/11/15/16e6cbdf53b0673a?w=953&h=426&f=png&s=309952)
複製代碼
這裏的a已經由基本數據類型,變成了對象類型。block會對對象類型的指針進行copy,copy到堆中,但並不會改變該指針所指向的堆中的地址,因此在上面的示例代碼中,block體內修改的實際是a指向的堆中的內容。
但若是咱們嘗試像上面圖片中的65行那樣作,結果會編譯不經過,那是由於此時你在修改的就不是堆中的內容,而是棧中的內容。
上文已經說過:Block不容許修改外部變量的值,這裏所說的外部變量的值,指的是棧中指針的內存地址。棧區是紅燈區,堆區纔是綠燈區。
系統的某些block api中,UIView的block版本寫動畫時不須要考慮,但也有一些api 須要考慮:
所謂「引用循環」是指雙向的強引用,因此那些「單向的強引用」(block 強引用 self )沒有問題,好比這些:
[UIView animateWithDuration:duration animations:^{ [self.superview layoutIfNeeded]; }];
複製代碼
[[NSOperationQueue mainQueue] addOperationWithBlock:^{ self.someProperty = xyz; }];
複製代碼
[[NSNotificationCenter defaultCenter] addObserverForName:@"someNotification"
object:nil
queue:[NSOperationQueue mainQueue]
usingBlock:^(NSNotification * notification) {
self.someProperty = xyz; }];
複製代碼
這些狀況不須要考慮「引用循環」。
但若是你使用一些參數中可能含有 ivar 的系統 api ,如 GCD 、NSNotificationCenter就要當心一點:好比GCD 內部若是引用了 self,並且 GCD 的其餘參數是 ivar,則要考慮到循環引用:
__weak __typeof__(self) weakSelf = self;
dispatch_group_async(_operationsGroup, _operationsQueue, ^
{
__typeof__(self) strongSelf = weakSelf;
[strongSelf doSomething];
[strongSelf doSomethingElse];
} );
複製代碼
相似的:
__weak __typeof__(self) weakSelf = self;
_observer = [[NSNotificationCenter defaultCenter] addObserverForName:@"testKey"
object:nil
queue:nil
usingBlock:^(NSNotification *note) {
__typeof__(self) strongSelf = weakSelf;
[strongSelf dismissModalViewControllerAnimated:YES];
}];
複製代碼
self --> _observer --> block --> self 顯然這也是一個循環引用。
檢測代碼中是否存在循環引用問題,可以使用 Facebook 開源的一個檢測工具 FBRetainCycleDetector 。
使用Dispatch Group追加block到Global Group Queue,這些block若是所有執行完畢,就會執行Main Dispatch Queue中的結束處理的block。
dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);
dispatch_group_t group = dispatch_group_create();
dispatch_group_async(group, queue, ^{ /*加載圖片1 */ });
dispatch_group_async(group, queue, ^{ /*加載圖片2 */ });
dispatch_group_async(group, queue, ^{ /*加載圖片3 */ });
dispatch_group_notify(group, dispatch_get_main_queue(), ^{
// 合併圖片
});
複製代碼
在並行隊列中,爲了保持某些任務的順序,須要等待一些任務完成後才能繼續進行,使用 barrier 來等待以前任務完成,避免數據競爭等問題。 dispatch_barrier_async
函數會等待追加到Concurrent Dispatch Queue並行隊列中的操做所有執行完以後,而後再執行 dispatch_barrier_async
函數追加的處理,等dispatch_barrier_async
追加的處理執行結束以後,Concurrent Dispatch Queue才恢復以前的動做繼續執行。
打個比方:好比大家公司週末跟團旅遊,高速休息站上,司機說:你們都去上廁所,速戰速決,上完廁所就上高速。超大的公共廁所,你們同時去,程序猿很快就結束了,但程序媛就可能會慢一些,即便你第一個回來,司機也不會出發,司機要等待全部人都回來後,才能出發。 dispatch_barrier_async
函數追加的內容就如同 「上完廁所就上高速」這個動做。
(注意:使用 dispatch_barrier_async
,該函數只能搭配自定義並行隊列 dispatch_queue_t
使用。不能使用: dispatch_get_global_queue
,不然 dispatch_barrier_async
的做用會和 dispatch_async
的做用如出一轍。 )
dispatch_get_current_queue
函數的行爲經常與開發者所預期的不一樣。 因爲派發隊列是按層級來組織的,這意味着排在某條隊列中的塊會在其上級隊列裏執行。 隊列間的層級關係會致使檢查當前隊列是否爲執行同步派發所用的隊列這種方法並不老是奏效。dispatch_get_current_queue
函數一般會被用於解決由不能夠重入的代碼所引起的死鎖,而後能用此函數解決的問題,一般也能夠用"隊列特定數據"來解決。
- (void)viewDidLoad
{
[super viewDidLoad];
NSLog(@"1");
dispatch_sync(dispatch_get_main_queue(), ^{
NSLog(@"2");
});
NSLog(@"3");
}
複製代碼
只輸出:1 。發生主線程鎖死。
// 添加鍵值觀察
/*
1 觀察者,負責處理監聽事件的對象
2 觀察的屬性
3 觀察的選項
4 上下文
*/
[self.person addObserver:self forKeyPath:@"name" options:NSKeyValueObservingOptionNew | NSKeyValueObservingOptionOld context:@"Person Name"];
複製代碼
observer中須要實現一下方法:
// 全部的 kvo 監聽到事件,都會調用此方法
/*
1. 觀察的屬性
2. 觀察的對象
3. change 屬性變化字典(新/舊)
4. 上下文,與監聽的時候傳遞的一致
*/
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context;
複製代碼
所謂的「手動觸發」是區別於「自動觸發」:
自動觸發是指相似這種場景:在註冊 KVO 以前設置一個初始值,註冊以後,設置一個不同的值,就能夠觸發了。
想知道如何手動觸發,必須知道自動觸發 KVO 的原理:
鍵值觀察通知依賴於 NSObject 的兩個方法: willChangeValueForKey:
和 didChangevlueForKey:
。在一個被觀察屬性發生改變以前, willChangeValueForKey:
必定會被調用,這就 會記錄舊的值。而當改變發生後, observeValueForKey:ofObject:change:context:
會被調用,繼而 didChangeValueForKey:
也會被調用。若是能夠手動實現這些調用,就能夠實現「手動觸發」了。
那麼「手動觸發」的使用場景是什麼?通常咱們只在但願能控制「回調的調用時機」時纔會這麼作。
具體作法以下:
若是這個 value
是 表示時間的 self.now
,那麼代碼以下:最後兩行代碼缺一不可。
相關代碼已放在倉庫裏。
// .m文件
// Created by https://github.com/ChenYilong
// 微博@iOS程序犭袁(http://weibo.com/luohanchenyilong/).
// 手動觸發 value 的KVO,最後兩行代碼缺一不可。
//@property (nonatomic, strong) NSDate *now;
- (void)viewDidLoad {
[super viewDidLoad];
_now = [NSDate date];
[self addObserver:self forKeyPath:@"now" options:NSKeyValueObservingOptionNew context:nil];
NSLog(@"1");
[self willChangeValueForKey:@"now"]; // 「手動觸發self.now的KVO」,必寫。
NSLog(@"2");
[self didChangeValueForKey:@"now"]; // 「手動觸發self.now的KVO」,必寫。
NSLog(@"4");
}
複製代碼
可是平時咱們通常不會這麼幹,咱們都是等系統去「自動觸發」。「自動觸發」的實現原理:
好比調用
setNow:
時,系統還會以某種方式在中間插入wilChangeValueForKey:
、didChangeValueForKey:
和observeValueForKeyPath:ofObject:change:context:
的調用。
你們可能覺得這是由於 setNow:
是合成方法,有時候咱們也能看到有人這麼寫代碼:
- (void)setNow:(NSDate *)aDate {
[self willChangeValueForKey:@"now"]; // 沒有必要
_now = aDate;
[self didChangeValueForKey:@"now"];// 沒有必要
}
複製代碼
這徹底沒有必要,不要這麼作,這樣的話,KVO代碼會被調用兩次。KVO在調用存取方法以前老是調用 willChangeValueForKey:
,以後老是調用 didChangeValueForkey:
。怎麼作到的呢?答案是經過 isa 混寫(isa-swizzling)。下文《apple用什麼方式實現對一個對象的KVO?》會有詳述。
參考連接: Manual Change Notification---Apple 官方文檔
均可以。
KVC 支持實例變量,KVO 只能手動支持手動設定實例變量的KVO實現監聽
請參考:
Apple 的文檔對 KVO 實現的描述:
Automatic key-value observing is implemented using a technique called isa-swizzling... When an observer is registered for an attribute of an object the isa pointer of the observed object is modified, pointing to an intermediate class rather than at the true class ...
從Apple 的文檔能夠看出:Apple 並不但願過多暴露 KVO 的實現細節。不過,要是藉助 runtime 提供的方法去深刻挖掘,全部被掩蓋的細節都會原形畢露:
當你觀察一個對象時,一個新的類會被動態建立。這個類繼承自該對象的本來的類,並重寫了被觀察屬性的 setter 方法。重寫的 setter 方法會負責在調用原 setter 方法以前和以後,通知全部觀察對象:值的更改。最後經過
isa 混寫(isa-swizzling)
把這個對象的 isa 指針 ( isa 指針告訴 Runtime 系統這個對象的類是什麼 ) 指向這個新建立的子類,對象就神奇的變成了新建立的子類的實例。我畫了一張示意圖,以下所示:
KVO 確實有點黑魔法:
Apple 使用了
isa 混寫(isa-swizzling)
來實現 KVO 。
下面作下詳細解釋:
鍵值觀察通知依賴於 NSObject 的兩個方法:willChangeValueForKey:
和 didChangevlueForKey:
。在一個被觀察屬性發生改變以前, willChangeValueForKey:
必定會被調用,這就會記錄舊的值。而當改變發生後, observeValueForKey:ofObject:change:context:
會被調用,繼而 didChangeValueForKey:
也會被調用。能夠手動實現這些調用,但不多有人這麼作。通常咱們只在但願能控制回調的調用時機時纔會這麼作。大部分狀況下,改變通知會自動調用。
好比調用 setNow:
時,系統還會以某種方式在中間插入 wilChangeValueForKey:
、 didChangeValueForKey:
和 observeValueForKeyPath:ofObject:change:context:
的調用。你們可能覺得這是由於 setNow:
是合成方法,有時候咱們也能看到有人這麼寫代碼:
- (void)setNow:(NSDate *)aDate {
[self willChangeValueForKey:@"now"]; // 沒有必要
_now = aDate;
[self didChangeValueForKey:@"now"];// 沒有必要
}
複製代碼
這徹底沒有必要,不要這麼作,這樣的話,KVO代碼會被調用兩次。KVO在調用存取方法以前老是調用 willChangeValueForKey:
,以後老是調用 didChangeValueForkey:
。怎麼作到的呢?答案是經過isa 混寫(isa-swizzling
)。第一次對一個對象調用 addObserver:forKeyPath:options:context:
時,框架會建立這個類的新的 KVO 子類,並將被觀察對象轉換爲新子類的對象。在這個 KVO 特殊子類中, Cocoa 建立觀察屬性的 setter ,大體工做原理以下:
- (void)setNow:(NSDate *)aDate {
[self willChangeValueForKey:@"now"];
[super setValue:aDate forKey:@"now"];
[self didChangeValueForKey:@"now"];
}
複製代碼
這種繼承和方法注入是在運行時而不是編譯時實現的。這就是正確命名如此重要的緣由。只有在使用KVC命名約定時,KVO才能作到這一點。
KVO 在實現中經過isa 混寫(isa-swizzling)
把這個對象的 isa 指針 ( isa 指針告訴 Runtime 系統這個對象的類是什麼 ) 指向這個新建立的子類,對象就神奇的變成了新建立的子類的實例。這在Apple 的文檔能夠獲得印證:
Automatic key-value observing is implemented using a technique called isa-swizzling... When an observer is registered for an attribute of an object the isa pointer of the observed object is modified, pointing to an intermediate class rather than at the true class ...
然而 KVO 在實現中使用了 isa 混寫( isa-swizzling)
,這個的確不是很容易發現:Apple 還重寫、覆蓋了 -class 方法並返回原來的類。 企圖欺騙咱們:這個類沒有變,就是本來那個類。。。
可是,假設「被監聽的對象」的類對象是 MYClass
,有時候咱們能看到對 NSKVONotifying_MYClass
的引用而不是對 MYClass
的引用。藉此咱們得以知道 Apple 使用了isa 混寫(isa-swizzling
)。具體探究過程可參考 這篇博文 。
那麼 wilChangeValueForKey:
、 didChangeValueForKey:
和 observeValueForKeyPath:ofObject:change:context:
這三個方法的執行順序是怎樣的呢?
wilChangeValueForKey:
、 didChangeValueForKey:
很好理解,observeValueForKeyPath:ofObject:change:context:
的執行時機是何時呢?
先看一個例子:
代碼已放在倉庫裏。
- (void)viewDidLoad {
[super viewDidLoad];
[self addObserver:self forKeyPath:@"now" options:NSKeyValueObservingOptionNew context:nil];
NSLog(@"1");
[self willChangeValueForKey:@"now"]; // 「手動觸發self.now的KVO」,必寫。
NSLog(@"2");
[self didChangeValueForKey:@"now"]; // 「手動觸發self.now的KVO」,必寫。
NSLog(@"4");
}
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSString *,id> *)change context:(void *)context {
NSLog(@"3");
}
複製代碼
若是單單從下面這個例子的打印上,
順序彷佛是 wilChangeValueForKey:
、 observeValueForKeyPath:ofObject:change:context:
、 didChangeValueForKey:
。
其實否則,這裏有一個 observeValueForKeyPath:ofObject:change:context:
, 和 didChangeValueForKey:
到底誰先調用的問題:若是 observeValueForKeyPath:ofObject:change:context:
是在 didChangeValueForKey:
內部觸發的操做呢? 那麼順序就是: wilChangeValueForKey:
、 didChangeValueForKey:
和 observeValueForKeyPath:ofObject:change:context:
不信你把 didChangeValueForKey:
注視掉,看下 observeValueForKeyPath:ofObject:change:context:
會不會執行。
瞭解到這一點很重要,正如 45. 如何手動觸發一個value的KVO 所說的:
「手動觸發」的使用場景是什麼?通常咱們只在但願能控制「回調的調用時機」時纔會這麼作。
而「回調的調用時機」就是在你調用 didChangeValueForKey:
方法時。
參考連接: Should IBOutlets be strong or weak under ARC?
文章告訴咱們:
由於既然有外鏈那麼視圖在xib或者storyboard中確定存在,視圖已經對它有一個強引用了。
不過這個回答漏了個重要知識,使用storyboard(xib不行)建立的vc,會有一個叫_topLevelObjectsToKeepAliveFromStoryboard的私有數組強引用全部top level的對象,因此這時即使outlet聲明成weak也不要緊
它可以經過KVC的方式配置一些你在interface builder 中不能配置的屬性。當你但願在IB中做盡量多得事情,這個特性可以幫助你編寫更加輕量級的viewcontroller
更多 lldb(gdb) 調試命令可查看