我們都知道在防止如block的循環引用時,會使用__weak關鍵字做如下定義:
__weaktypeof(self) weakSelf =self;
后來,為了方便,不用每次都要寫這樣一句固定代碼,我們定義了宏:
#defineWeakSelf__weak typeof(self) weakSelf =self;
之后,我們可以比較方便的在需要的地方:
WeakSelf;...[weakSelf doSomething];
再后來,我們發現不止self需要使用weak,可能有部分變量也需要weak,于是我們的宏繼續進化,不僅僅只支持self:
#defineWeakObj(o) __weak typeof(o) o##Weak = o;
這樣,后續對需要使用weak的對象,只要寫一句WeakObj(obj)即可使用objWeak變量了(PS:發現沒有,這里生成的變量名其實是objWeak,并不是weakObj,原因見文章末的注1)
再后來,我們發現了一些小技巧,可以讓我們的這個宏看起來更原生一些,我們添加了@符號在前面:
#defineWeakObj(o) autoreleasepool{} __weak typeof(o) o##Weak = o;
使用上看起來是這樣
@WeakObj(self);...[selfWeak doSomething];
是不是感覺挺高大上的?
這里是利用了@autoreleasepool{}這個系統的關鍵字來實現的,其實還可以利用@try{}@finally{}這個也可以達到相同的效果,比如:
#defineWeakObj(o)try{}@finally{}__weaktypeof(o)o##Weak = o;
這部分空的@try或者空的@autoreleasepool會在編譯時被優化掉,不必擔心性能問題。
至此,我們的宏已經可以用了,但是實際使用中,出現了一個很尷尬的問題,就是代碼自動補全,@W并不能自動提示出該宏,所以每次都是很尷尬的先利用提示,寫完WeakObj(obj),然后光標移動到前面去打上一個@符號。
這種事情怎么能忍受?
還好我們還有利器,Xcode的CodeSnippet,任意方法內,寫一句代碼
@WeakObj(<#obj#>);
拖到Xcode的CodeSnippet區域,快捷鍵設置為@WeakObj即可。
至此,愉快的使用@W即可自動補全出該宏了。。
另外,還有相應的strong宏,一并曬在這里
#defineStrongObj(o) autoreleasepool{} __strong typeof(o) o = o##Weak;
用處嘛簡單寫個例子:
@WeakObj(self);
[varsetBlock:^{
? ? ? ?@StrongObj(self);? ??
? ? ? ?[self doSomething];
}];
關于這么寫的原因,請先自行揣摩,之后可以看看曉月的這篇文章:http://blog.csdn.net/uxyheaven/article/details/44226395,也可以在評論中留言。
最后,揭曉為什么該宏生成的變量名是objWeak:
1. 使用時,如果開發者習慣性的要打出 [self doSomething]時,當他輸入self時,自動補全出來的部分能看到還有selfWeak可供選擇,算是一種提醒方式。
2. 如果weak前置,當然也可以,生成的會是weakobj這樣的變量名,只需要把宏中o##Weak換成weak##o
好了,這篇文章希望對一些人有些啟發或幫助。
最后曬出個人在用的宏定義:
#defineYRWeakObj(o) autoreleasepool{} __weak typeof(o) o##Weak = o;#defineYRStrongObj(o) autoreleasepool{} __strong typeof(o) o = o##Weak;
