objc runtime (五)属性及成员变量 1

属性和成员变量的区别

成员变量、属性

成员变量一般由_obj表示,默认为@protected一般类内访问使用。类外访问需要添加@public
属性一般由@property生成,对成员变量扩充了gettersetter,并且默认会生成带下划线的成员变量。
个人理解:属性一定是成员变量,成员变量不一定是属性。

注意:上一篇《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() 进行释放该数组。

获取成员变量和属性的名称

通过上面两个函数,我们了解到如何获得类型为Ivarobjc_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等成员变量及telID等属性。

@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

你可能感兴趣的:(objc runtime (五)属性及成员变量 1)