美文网首页iOS Developer
Objective-C 懒加载没有调用?

Objective-C 懒加载没有调用?

作者: 小時間光 | 来源:发表于2017-04-21 16:10 被阅读232次

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

    懒加载 亮点自己找

       个人推测,可能是写这个代码的哥们用了懒加载,然后发现给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.

    最终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出来的结果请看图:

    懒加载的正确打开方式

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


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

    本文demo请戳这里;

    本文参考了:   Encapsulating Data

                        《Objective-C编程全解》 

                        《Effective Objective-C 2.0》

    相关文章

      网友评论

        本文标题:Objective-C 懒加载没有调用?

        本文链接:https://www.haomeiwen.com/subject/gfheottx.html