懒加载
- 当我们用到的时候才加载,不用的时候不加载。其本质是重写getter方法。
- 懒加载的好处是:使代码的可读性更强,代码之间的独立性更强,松耦合,并且节省了内存资源。
什么是懒加载
懒加载是延时加载,即在使用到该对象的时候才将其加载到内存中。
为什么要用懒加载
主要目的是为了提升效率
- 在iOS开发中,加载数据的操作一般放在viewDidLoad中,而如果此时展示的界面不需要数据,那我们可能会因为人为的加载了数据而影响了系统的性能。还不如让系统利用这段时间做它认为更加重要的事情。
- 在iOS开发中,如果某个APP占用内存过大,系统会给这个APP发送内存警告,一般当APP接收到内存警告时,即在
didReceiveMemoryWarning
方法中程序员会做一些清理内存的操作,此时我们的数据或者其他控件占用的内存可能会被清理掉,这时如果再去访问这块内存就会出现野指针的问题。
-
综合以上两点,我们采用懒加载的方式对数据以及控件进行加载,这样可以保证在使用的时候去加载,而且能保证使用时肯定有。
怎么用懒加载
懒加载的实现,就是重写对象的getter方法。即在系统调用某个对象之前先进行判断,如果有这个对象(即对象不为空)就直接返回这个对象,如果对象为空就创建它并且返回它。一般而言需要懒加载的是数据类型和UI控件,以下将对这两点分开阐述。
数据类型的懒加载
数据类型的懒加载一般包括字典,数组,以及自定义的模型数据。以数组为例,懒加载的实现为:
@interface NewObject() //声明一个数组属性(数据类型声明为strong) @property (nonatomic, strong) NSArray * myArray; @end @implementation NewObject //重写myArray的getter方法 - (NSArray *)myArray{ if (_myArray == nil) { _myArray = [[NSArray alloc] init]; } return _myArray; }
UI控件的懒加载
UI控件一般包括UIView, UIButton等控件。以UIButton为例,如果使用storyboard,在连线时系统默认在类扩展中自动生成
@property (weak, nonatomic) IBOutlet UIButton *myButton;
。之所以声明为weak属性,是因为storyboard已经对myButton做了强引用,所以在类扩展中直接声明为weak属性,这样减少了出现内存泄露的可能。
而使用纯代码实现时,一般的代码模式是@property (weak, nonatomic) UIButton *myButton; @end @implementation ViewController - (UIButton *)myButton{ if (_myButton == nil) { UIButton * tempButton = [[UIButton alloc] init]; _myButton = tempButton; [self.view addSubview: _myButton]; } return _myButton; }
以上代码需要注意几个地方:
-
为了和storyboard的形式一致,同时也是为了让控件在不用的时候能及时被销毁,属性也声明成了weak类型。
-
weak类型的变量就不能使用
_myButton = [[UIButton alloc] init];
进行定义。因为myButton是用weak修饰的,这样用[[UIButton alloc] init]
申请的内存就只被weak类型的指针指向,而根据ARC内存管理机制,如果一个对象不再被strong指针指向,它将会被释放。所以此处要利用以上的格式定义这个对象。即
UIButton * tempButton = [[UIButton alloc] init]; _myButton = tempButton;
这样在本层大括号的作用域内申请的内存地址就有强指针指向了。 - 对于子控件必须要加载到父view上才能显示出来。所以必须执行
addSubview
,而这条语句在懒加载中实现即可。具体原因如以下记录中的注释
if (_myButton == nil) {
UIButton * tempButton = [[UIButton alloc] init];
_myButton = tempButton;
[self.view addSubview: _myButton];//调用addSubview,self.view 会对myButton有强引用
}//出了这个大括号,tempButton 这个局部对象将会被销毁,如果前面没有执行addSubview,那新申请的button内存又会被释放
return _myButton;//如果不执行addSubview或者其他对button内存强引用的语句,那么此处返回的将是nil。
-