属性和成员变量的区别
成员变量、属性
成员变量一般由_obj
表示,默认为@protected
一般类内访问使用。类外访问需要添加@public
。
属性一般由@property
生成,对成员变量扩充了getter
和setter
,并且默认会生成带下划线的成员变量。
个人理解:属性一定是成员变量,成员变量不一定是属性。
注意:上一篇《objc runtime (四)动态添加属性》中利用关联来添加的,严格来说不应被称之为属性(由于它没有生成带下划线的成员变量)。
获取成员变量和属性列表
函数说明
runtime中有class_copyIvarList()
和class_copyPropertyList()
两个方法:
/**
* Describes the instance variables declared by a class.
*
* @param cls The class to inspect.
* @param outCount On return, contains the length of the returned array.
* If outCount is NULL, the length is not returned.
*
* @return An array of pointers of type Ivar describing the instance variables declared by the class.
* Any instance variables declared by superclasses are not included. The array contains *outCount
* pointers followed by a NULL terminator. You must free the array with free().
*
* If the class declares no instance variables, or cls is Nil, NULL is returned and *outCount is 0.
*/
Ivar *class_copyIvarList(Class cls, unsigned int *outCount);
/**
* Describes the properties declared by a class.
*
* @param cls The class you want to inspect.
* @param outCount On return, contains the length of the returned array.
* If \e outCount is \c NULL, the length is not returned.
*
* @return An array of pointers of type \c objc_property_t describing the properties
* declared by the class. Any properties declared by superclasses are not included.
* The array contains \c *outCount pointers followed by a \c NULL terminator. You must free the array with \c free().
*
* If \e cls declares no properties, or \e cls is \c Nil, returns \c NULL and \c *outCount is \c 0.
*/
objc_property_t *class_copyPropertyList(Class cls, unsigned int *outCount);
从官方定义中的注释我们可以看出class_copyIvarList()
是用来描述类声明的实例变量
,class_copyPropertyList
是用来描述类声明的属性
。下面来说明参数的意义。
class_copyIvarList()
-
cls
是需要操作的类 -
*outCount
是函数返回的数组的长度。这里我们需要定义一个unsigned int
类型的变量,利用&
将定义的变量的地址传入。 - 这个函数会返回一个元素为
Ivar
类型的数组的指针,这个数组包含*outCount
个指针,并会以NULL
结束。使用之后必须使用free()
进行释放该数组。
class_copyPropertyList()
-
cls
是需要操作的类 -
*outCount
是函数返回的数组的长度。这里我们需要定义一个unsigned int
类型的变量,利用&
将定义的变量的地址传入。 - 这个函数会返回一个元素为
objc_property_t
类型的数组的指针,这个数组包含*outCount
个指针,并会以NULL
结束。使用之后必须使用free()
进行释放该数组。
获取成员变量和属性的名称
通过上面两个函数,我们了解到如何获得类型为Ivar
或objc_property_t
的成员变量。runtime为我们提供了两个方法可以得到成员变量及属性的名称。
/**
* Returns the name of an instance variable.
*
* @param v The instance variable you want to enquire about.
*
* @return A C string containing the instance variable's name.
*/
const char * _Nullable ivar_getName(Ivar _Nonnull v);
/**
* Returns the name of a property.
*
* @param property The property you want to inquire about.
*
* @return A C string containing the property's name.
*/
const char * _Nonnull property_getName(objc_property_t _Nonnull property);
我们可以了解,返回值类型为char *
。
具体实现
首先我们来定义一个类person
,包含_name
、_sex
、_age
、_nation
等成员变量及tel
、ID
等属性。
@interface Person : NSObject
{
NSString *_name;
NSString *_sex;
NSInteger _age;
NSString *_nation;
}
@property (nonatomic, copy) NSString *tel;
@property (nonatomic, copy) NSString *ID;
@end
我们再使用下面代码分别遍历Person
类的成员变量与属性。
unsigned int ivar_count;
Ivar *ivars = class_copyIvarList([Person class], &ivar_count);
NSLog(@"开始遍历成员变量:");
for (int i = 0; i < ivar_count; i++) {
Ivar ivar = ivars[i];
NSString *ivar_name = [NSString stringWithUTF8String:ivar_getName(ivar)];
NSLog(@"%@", ivar_name);
}
unsigned int property_count;
objc_property_t *properties = class_copyPropertyList([Person class], &property_count);
NSLog(@"开始遍历属性:");
for (int i = 0; i < property_count; i++) {
objc_property_t property = properties[i];
NSString *property_name = [NSString stringWithUTF8String:property_getName(property)];
NSLog(@"%@", property_name);
}
运行结果如下:
2018-11-12 23:46:59.228659+0800 runtimeDemo[1485:160439] 开始遍历成员变量:
2018-11-12 23:46:59.228908+0800 runtimeDemo[1485:160439] _name
2018-11-12 23:46:59.228987+0800 runtimeDemo[1485:160439] _sex
2018-11-12 23:46:59.229005+0800 runtimeDemo[1485:160439] _age
2018-11-12 23:46:59.229016+0800 runtimeDemo[1485:160439] _nation
2018-11-12 23:46:59.229024+0800 runtimeDemo[1485:160439] _tel
2018-11-12 23:46:59.229057+0800 runtimeDemo[1485:160439] _ID
2018-11-12 23:46:59.229088+0800 runtimeDemo[1485:160439] 开始遍历属性:
2018-11-12 23:46:59.229101+0800 runtimeDemo[1485:160439] tel
2018-11-12 23:46:59.229111+0800 runtimeDemo[1485:160439] ID
Program ended with exit code: 0