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