Objective-C 懒加载没有调用?

外包出去的项目,然后二期拿回来自己做,今天改BUG时发现了一个很有意思的地方:请看图:

《Objective-C 懒加载没有调用?》 懒加载

《Objective-C 懒加载没有调用?》 亮点自己找

   个人推测,可能是写这个代码的哥们用了懒加载,然后发现给dataArray数组赋值后,数组还是nil,所以在转Model之前又初始化了一次,至于原因请看我慢慢分析:

1、getter和setter方法:

property直译就是属性,一般我们用来保存和操作对象的数据。那么当我们用@property申明了一个属性后,Xcode为我们做了什么呢?

(1)、setter和getter的写法

在Objective-C中通常申明一个属性的写法为:

@property (strong, nonatomic) NSString *Data;

此时Xcode IDE已经自动的为开发者生成了getter方法(“读取”)和setter方法(“设置”),此时,默认这个属性是可读可修改的;

@property (strong, nonatomic, readwrite) NSString *Data;

以上两种申明属性产生的效果是相同的,虽然我们没有用readwrite关键字,但是Xcode已经为我们自动生成了这个属性的getter和setter方法;

为了近一步说明getter和setter方法,请看下面两段代码:

代码1:

@property (assign, nonatomic) int Numcode;

代码2:

-(int) Numcode;//getter 
-(void) setNumcode:(int)numcode;//setter

上面的两种写法产生的效果是一样的。从我的个人角度来说,初期的学习很多的教材上都使用的是上面的第一种代码,而且相对于第二种写法第一种写法也比较简便易懂,再加上大家都习惯了这种写法,所有在我的项目中我都使用的是第一种写法;

当然也不一定,在一些比较老的项目中会使用以下的方法来申明:

@property int age; //表示声明了一个属性和getter和setter
@synthesize age = _age; //表示实现setteer和getter

(2)、setter和getter的调用:

setter和getter的调用其实就是对属性的读取和赋值操作,不说废话直接上代码吧:

#import"EOCClass.h"@implementation EOCClass+(EOCClass*)sharedInstance{    static dispatch_once_t EOCClassperonce;    static EOCClass* eOCClass;    dispatch_once(&EOCClassperonce, ^{        eOCClass = [[EOCClass alloc]init];        eOCClass.Numcode = 10086;    });  return eOCClass;}
EOCClass *eClass = [EOCClass sharedInstance];//单利模式初始化一个类//--------------操作类属性-------------eClass.Numcode = 1008611;   [eClass setNumcode:1008611]; //setter方法调用,两种写法是等效的
int lsCode = eClass.Numcode; int lsNum = [eClass Numcode];//getter方法调用,两种写法是等效的

(3)、other

至于在申明属性时用到的一些关键字如:

atomic   nonatomic  readonly  readwirte   assgin  strong   weak  copy  retain

等等的区别这里就不做赘述了。

 2、self和下划线_的使用:

(1)、self和下划线_的区别

      1、 self.是调用属性的getter和setter方法,编译器在生成getter,setter方法时,编译器首先查找当前的类中用户是否定义属性的getter,setter方法,如果有,则编译器会跳过,不会再生成,使用用户定义的方法。

      2、使用 self.赋值操作的时候实际上会先调 release(就是retainCount -1)一次,然后再把指针指向这个属性,下划线_赋值时没有release操作,直接把指针指向这个属性的实例变量。

      3、 在使用self.时是调用一个getter和setter方法。会使引用计数加一当这个属性在调用下划线_是直接对属性的实例变量进行操作;

      4、下划线_Numcode实际上和self->_Numcode;的效果是相同的;

     5、 下划线_Numcode只能获取局部变量,不能获取到父类的属性或方法;

请看下面的代码:

-(void)DatafromServer{   //--------------------   //setter   self.Numcode = 1008611;   //getter   int digital = self.Numcode;   //getter   int digitaltoo = [self Numcode];   printf("\\DatafromServer=========%d,%d\n",digital,digitaltoo);}
-(int)DatafromServertoo{    //--------------------   //setter   [self setNumcode:1008611];    //getter   int digital = self.Numcode;   //getter   int digitaltoo = [self Numcode];   printf("\nDatafromServertoo=========%d,%d\n",digital,digitaltoo);}
EOCClass*eClass = [EOCClasssharedInstance];[eClassDatafromServer];[eClassDatafromServertoo];,

最后可以看到DatafromServer方法Log值和DatafromServertoo方法Log值都是1008611.

《Objective-C 懒加载没有调用?》 最终Log值

总结:self.对属性的get和set方法间接调用,下划线_是直接对实例变量操作。

3、懒加载:

前面已经说到,在iOS 5之后,使用`@property`定义一个属性后,系统会默认生成`getter`和`setter`方法。我们申明了一个属性,但并不是立即就要使用这个对象,没必要把所有的属性都放在`viewDidLoad`方法中初始化,等到要使用时再加载(初始化)。

使用懒加载需要注意:

**

(1)、当开发者使用懒加载本质就是重写了getter()方法;

(2)、懒加载在加载时必须判空;

(3)、懒加载判空必须使用下划线,如下面代码,Xcode100%会报错的,前面已经说明,self就是调用了setter跟getter方法,懒加载本质就是重写了getter方法,但在此处属性本身还没初始化,是nil,但是getter返回的也是nil,那在判断时就会进入死循环;

**

//错误的懒加载示范一-(NSArray*)PageddatafromServerList{   if(self.PageddatafromServerList == nil)   {      self.PageddatafromServerList = [NSArray array];    }  return self.PageddatafromServerList;}
//错误的懒加载示范二-(NSArray*)PageddatafromServerList{   if(self.PageddatafromServerList) //此处一定要判空   {     self.PageddatafromServerList = [NSArray array];    }  return self.PageddatafromServerList;}
//正确的懒加载方式-(NSArray *)DepartmentArray{   if(_DepartmentArray == nil)   {      _DepartmentArray = [NSArray array];   }return _DepartmentArray;}

分析:最上面贴出的代码出现的原因:

  由于这个写这个代码的哥们使用了懒加载,而懒加载本质上是重写了属性的getter方法,本文第二条也说明了self.和_的区别,所以在赋值时使用_dataArray,就没有调dataArray的getter方法,懒加载根本就没有调!!!所以出现的情况就是给dataArray赋值后依旧是nil!

   直接就举个🌰来说明吧:我买了个超级省电的台灯,回家后我给台灯通上电后发现怎么折腾这个台灯都不亮,为什么呢?因为我没有摁台灯的开关!!

那么懒加载的正确打开方式是怎样的呢?请看代码:

@interface EOCClass :NSObject@property (strong, nonatomic) NSArray *LazyLoading;-(void)LazyLoadingData;@end
-(void)LazyLoadingData{
    NSLog(@"_LazyLoading============%@",_LazyLoading);    NSLog(@"self.LazyLoading=============%@",self.LazyLoading);    NSLog(@"=============%@",[self LazyLoading]);}
-(NSArray*)LazyLoading{   if(!_LazyLoading)   {     _LazyLoading= [NSArray array];   }  return _LazyLoading;}

最终Log出来的结果请看图:

《Objective-C 懒加载没有调用?》 懒加载的正确打开方式

那么,通过上面的代码,可以看出,使用了懒加载后,当要使用这个对象时可以用self.调用该对象或者直接调用这个对象的getter方法,如果用下划线调用实例变量那么懒加载就没有调用,最终造成的结果就是赋值了也是nil;

以上是个人的一些理解和总结,如果有错误的地方请指出。

本文demo请戳这里;

本文参考了:   Encapsulating Data

                    《Objective-C编程全解》 

                    《Effective Objective-C 2.0》

    原文作者:小時間光
    原文地址: https://www.jianshu.com/p/986adc6fe3ca
    本文转自网络文章,转载此文章仅为分享知识,如有侵权,请联系博主进行删除。
点赞