iOS-属性与实例变量(成员变量)
在我看来iOS开发中属性和实例变量是两个概念,还是有一定区别的:
1. 只声明实例变量,类外部不可访问,不可赋值,类内部可以通过self->属性名或者属性名访问和赋值
2. 声明并实现实例变量的setter方法和getter方法,类外部和内部都可以通过类.实例变量名(本质上就是调用实例变量的setter方法和getter方法)访问和赋值
2.1 iOS5之前为了避免麻烦声明实例变量的setter和getter方法,可通过@property声明属性对应声明实例变量,
@property相当于声明了实例变量的setter方法和getter方法,成对出现的@synthesize相当于实现实例变量的setter方法和getter方法
2.2 iOS5之后系统推荐使用如下用法,编译器会自动生成一个名为_属性名的实例变量与属性相对应(在没有写@synthesize的情况下)
注意:如果你添加了@synthesize 属性名 那么_属性名就不存在了,声明和实现的就是名为属性名的实例变量的setter和getter方法
如果.m文件中需要同时重写实例变量的setter和getter方法时,在方法里使用_属性名会报错,因为你重写实例变量的setter和getter方法后,_属性名就不存在了和添加@synthesize同一个道理,你可以添加@synthesize 属性 = _属性 这样你就可以在setter和getter方法中使用_属性名了,或者重写_属性的setter和getter方法(如下,不推荐使用)
3.category中的属性和成员变量
面试的时候经常会被问到“类别可以扩展属性嘛?”,面对这个问题不同工作年限的人回答的不一样。刚刚从事iOS的人可能立刻回答不能,然后面试官很诡异的偷偷一笑就继续下面一个面试题了,很明显你要是这么回答你就可能已经掉坑里了😂,对类别有一定了解的人就会了解,类别并不是不可以扩展属性的,只不过类别中不可以添加成员变量(实例变量)而已。
3.1 类别中添加@property描述的属性,这时候编译是没问题的没警告没报错,但是一旦在创建对象后为属性赋值或者使用这个属性的时候,程序就崩溃了,奔溃的原因也很简单,就是找不到属性的set/get方法。
那我们就按照这个流程来,在类别中为属性添加set/get方法,在set方法里面赋值的时候找不到赋值的对象,也就是说系统没有为我们生成带下划线的成员变量,没生成我们就自己加,😂一加就报错。看来这才是类别不能扩展属性的根本原因。
但是办法都是人想出来的嘛!不能用实例变量的话我就用全局变量保存属性的值就好了,没想到还真可以,创建对象后可以赋值,可以获取值。
利用runtime里面的class_copyPropertyList()获取类的所有property,这个属性居然也在里面,但是使用class_copyIvarList()的时候却没有对应的成员变量,也就是说针对这个属性,系统没有生成对应的成员变量。当然这种方法只是用来学习的,真正开发中估计会用到另一种办法:利用runtime的关联对象。
runtime中的关联对象说的简单点就是利用键值对的方式为两个对象建立关联关系,详细的资料可以查看iOS runtime 关联对象。
3.2 类别中添加成员变量,这种方式XCode直接给你报错了:Instance Value may not be placed in the categories。很直接了当的告诉你类别中不能放置实例变量,至于原因就需要从runtime中去了解了,我也只是了解了皮毛,至于细节可以查看runtime详解。