外包出去的项目,然后二期拿回来自己做,今天改BUG时发现了一个很有意思的地方:请看图:
个人推测,可能是写这个代码的哥们用了懒加载,然后发现给dataArray数组赋值后,数组还是nil,所以在转Model之前又初始化了一次,至于原因请看我慢慢分析:
1、getter和setter方法:
property直译就是属性,一般我们用来保存和操作对象的数据。那么当我们用@property申明了一个属性后,Xcode为我们做了什么呢?
(1)、setter和getter的写法
在Objective-C中通常申明一个属性的写法为:
```
@property (strong, nonatomic) NSString *Data;
```
@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.
总结: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