回憶 oc 的內(nèi)存管理:
objective-c 語法快速過(6)內(nèi)存管理原理,objective-c 語法快速過(7)編譯器特性ARC
ARC是蘋果為了簡(jiǎn)化程序員對(duì)內(nèi)存的管理,推出的一套內(nèi)存管理機(jī)制,使用ARC機(jī)制,對(duì)象的申請(qǐng)和釋放工作會(huì)在運(yùn)行時(shí),由編譯器自動(dòng)在代碼中添加retain和release
1> strong:強(qiáng)指針引用的對(duì)象,在生命周期內(nèi)不會(huì)被系統(tǒng)釋放,在OC中,對(duì)象默認(rèn)都是強(qiáng)指針
2> weak:弱指針引用的對(duì)象,系統(tǒng)會(huì)立即釋放,弱指針可以指向其他已經(jīng)被強(qiáng)指針引用的對(duì)象
他們都是 arc 的東西
ARC的判斷準(zhǔn)則:
只要沒有強(qiáng)指針指向?qū)ο螅蜁?huì)釋放對(duì)象,弱指針不會(huì)這樣,及時(shí)有弱指針指向?qū)ο螅瑢?duì)象沒有強(qiáng)指針指向,也會(huì)自動(dòng)釋放掉。一般,無需顯式聲明為強(qiáng)指針,但是在封裝里,定義方法的時(shí)候需要寫明。而弱指針,必須顯式說明。默認(rèn)是強(qiáng)指針。
ARC特點(diǎn)
1> 不允許調(diào)用release、retain、retainCount
2> 允許重寫dealloc,但是不允許調(diào)用[super dealloc]
3> @PRoperty的參數(shù)
* strong :成員變量是強(qiáng)指針(適用于OC對(duì)象類型)
* weak :成員變量是弱指針(適用于OC對(duì)象類型)
* assign : 適用于非OC對(duì)象類型
4> 以前的retain改為用strong
oc的指針分2種:
1> 強(qiáng)指針:默認(rèn)情況下,所有的指針都是強(qiáng)指針 __strong
2> 弱指針:__weak
在 點(diǎn)m 文件
#import "Dashuai.h"@implementation Dashuai- (void)dealloc{ NSLog(@"對(duì)象被銷毀了!"); //[super dealloc];}@end
在視圖控制器
@interface ViewController ()@property (nonatomic, strong) Dashuai *da;@end@implementation ViewController- (void)viewDidLoad { [super viewDidLoad]; self.da = [[Dashuai alloc] init]; NSLog(@"調(diào)用了 viewdidlowd 方法!");}@end
對(duì)象不銷毀,da 是強(qiáng)指針對(duì)象。
如果是在方法內(nèi)部,聲明的對(duì)象(默認(rèn)都是強(qiáng)指針),那么存在一個(gè)聲明周期(局部變量)的問題。方法執(zhí)行完畢,到關(guān)括號(hào)完畢,內(nèi)存也就隨機(jī)自動(dòng)銷毀了。
- (void)viewDidLoad { [super viewDidLoad]; //self.da = [[Dashuai alloc] init]; Dashuai *dashuai = [[Dashuai alloc] init]; NSLog(@"調(diào)用了 viewdidlowd 方法!");}
2015-03-08 17:12:33.240 strong和 weak[20017:738240] 調(diào)用了 viewdidlowd 方法!
2015-03-08 17:12:33.241 strong和 weak[20017:738240] 對(duì)象被銷毀了!
如果換__weak,那么語句執(zhí)行完畢,立即釋放內(nèi)存,加斷點(diǎn)課證明
2015-03-08 17:15:41.480 strong和 weak[20062:739972] 對(duì)象被銷毀了!
(lldb)
下面看堆棧圖
上例,強(qiáng)指針的話,那么指針變量dashuai在內(nèi)存的棧區(qū),僅僅是內(nèi)存的地址,指向的內(nèi)存在堆區(qū),
等指針不再指向這塊內(nèi)存,或者聲明周期結(jié)束,內(nèi)存才釋放
對(duì)于弱指針指向,是虛線
執(zhí)行之后,對(duì)象內(nèi)存立即被釋放。
再看控件之間,如圖:
視圖控制器強(qiáng)引用 view,view 強(qiáng)引用了子控件。
[self.view addSubView];//讓 view 對(duì)控件強(qiáng)引用
那么這就是以前寫代碼的時(shí)候,連線故事板的控件和代碼關(guān)聯(lián)的時(shí)候,總是默認(rèn)weak的原因。因?yàn)槟鞘且晥D控制器在引用子控件,而 view 已經(jīng) strong 了,那么視圖控制器就沒必要 strong。它的內(nèi)存釋放過程是:
如果程序沒有必要弄強(qiáng)引用,那么就用若引用,類比是鏈表,一條龍,視圖和姿勢(shì)圖,只一個(gè)實(shí)線連接(父類對(duì)象,強(qiáng)引用子類對(duì)象之后),其余的引用用 weak虛線,因?yàn)?span id="70qarhlowxag" class="s2">view 已經(jīng)對(duì)子控件強(qiáng)引用了,視圖控制器可以不用 strong。這是蘋果的應(yīng)用內(nèi)存管理機(jī)制。當(dāng)然用 strong 也不是不可以。
還有之前提到的懶加載
控件的懶加載,是手動(dòng)的寫類屬性的 set 方法,進(jìn)行是否已經(jīng)加載等的判斷,避免重復(fù)加載的過程。那么手寫 set ,在 viewdidload重寫代碼,就要用 strong 引用,因?yàn)槭鞘謱懘a,那么如果還是 weak ,在 viewdidload 方法里,使用控件對(duì)象,那么一旦執(zhí)行完畢,立即內(nèi)存被釋放,因?yàn)槭?weak 的。這樣在視圖加載之前,對(duì)象就已經(jīng)消失了。
注意下面的方法調(diào)用時(shí)間
init方法-初始化程序viewDidLoad方法-加載視圖viewWillAppear方法在,UIViewController對(duì)象的視圖即將加入窗口時(shí)調(diào)用;
viewDidApper方法在UIViewController對(duì)象的視圖已經(jīng)加入到窗口時(shí)調(diào)用;
viewWillDisappear方法咋UIViewController對(duì)象的視圖即將消失、被覆蓋或是隱藏時(shí)調(diào)用;
viewDidDisappear方法在UIViewController對(duì)象的視圖已經(jīng)消失、被覆蓋或是隱藏時(shí)調(diào)用;
didReceiveMemoryWarning -在內(nèi)存不足時(shí)候調(diào)用
@property參數(shù)使用小結(jié):
1> 控件用weak
2> 屬性對(duì)象用strong
3> 非對(duì)象類型用assign
4> 字符串NSString用copy(不是可變字符串)
提示:在純手碼實(shí)現(xiàn)界面布局時(shí),如果通過懶加載處理界面控件,需要使用strong強(qiáng)指針
新聞熱點(diǎn)
疑難解答
圖片精選
網(wǎng)友關(guān)注