在runtime.h中,你可以通过其中的class_copyIvarList方法来获取实例变量。具体的实现如下(记得导入头文件<objc/runtime.h>):

- (NSArray *)ivarArray:(Class)cls {
unsigned int stuIvarCount = 0;
Ivar *ivars = class_copyIvarList(cls, &stuIvarCount);
if (stuIvarCount == 0) {
return nil;
}
NSMutableArray *arr = [[NSMutableArray alloc] initWithCapacity:stuIvarCount];
for (int i = 0;i<stuIvarCount;i++) {
Ivar ivar = ivars[i];
NSString *ivarName = [NSString stringWithUTF8String:ivar_getName(ivar)];
NSLog(@"%@",ivarName);
[arr addObject:ivarName];
}
free(ivars);
return arr;
}

如上面代码。其中cls就是你要获取实例变量的类,stuIvarCount用来承载要获取类的实例变量的个数。打印出来的ivarName就是cls的实例变量。接下来对这个方法进行解析:

首先看一下里面的Ivar,先看一下定义:

/// An opaque type that represents an instance variable.
typedef struct objc_ivar *Ivar; struct objc_ivar {
char * _Nullable ivar_name OBJC2_UNAVAILABLE; //变量名字
char * _Nullable ivar_type OBJC2_UNAVAILABLE; //变量类型
int ivar_offset OBJC2_UNAVAILABLE; //偏移量
#ifdef __LP64__
int space OBJC2_UNAVAILABLE; //存储空间
#endif
}

Ivar是一个叫做objc_ivar的结构体指针,其中的 ifdef判断是判断当前设备是否是64位设备,这里可以延伸出一个方法:

//判断当前设备是否是64位设备,也可以用这个方法判断是否是32位设备
- (BOOL)is64Bit {
#if defined(__LP64__) && __LP64__
return YES;
#else
return NO;
#endif
}
OBJC_EXPORT Ivar _Nonnull * _Nullable
class_copyIvarList(Class _Nullable cls, unsigned int * _Nullable outCount)
OBJC_AVAILABLE(10.5, 2.0, 9.0, 1.0, 2.0);

class_copyIvarList的注释如下:



它返回的是一个Ivar的数组,这个数组里面包含了你要查看类的所有实例变量,但是不包括从父类继承过来的。如果你传入的类没有实例变量或者改class为Nil,那么该方法返回的就是NULL,count值也就变成了0。有一点需要注意:你必须使用free()方法将该数组释放

然后就是通过for循环遍历,通过ivar _ getName拿到ivarName。

以上便是对clas_copyIvarList的介绍。

它还有一个最常用的使用方式(在开发中经常用到的):根据字典或者json字符串转化为model,在网络请求返回数据时经常用到。使用方法就是自己写一个基类的model,然后让项目中用到的model都继承自此基类,基类中的关键代码如下:

+ (instancetype)zg_modelFromDic:(NSDictionary *)dataDic {
id model = [[self alloc] init];
unsigned int count = 0; Ivar *ivarsA = class_copyIvarList(self, &count);
if (count == 0) {
return model;
}
for (int i = 0;i < count; i++) {
Ivar iv = ivarsA[i];
NSString *ivarName = [NSString stringWithUTF8String:ivar_getName(iv)];
ivarName = [ivarName substringFromIndex:1];
id value = dataDic[ivarName];
[model setValue:value forKey:ivarName];
}
free(ivarsA);
return model;
}

这里是把字典转成model,先用class_copyIvar获取该model的所有实例变量,然后通过kvc对属性进行赋值。最终返回model。这里有个点需要注意以下几点:

  1. 你的model的属性名称要和服务端返回的数据一致,比如你的model有个属性叫做name,那么你服务端返回的数据字典里面的对应属性也要叫做name,因为这个方法是根据属性从字典里面拿数据的。你也可以做一个映射,让自定义的实例变量名称映射到服务端提供的变量名称。
  2. 实现里面有个substringFromIndex:操作,其目的就是把使用该方法拿到的实例变量前面的" _ "去掉。所以你最好使用 @property 进行属性声明,并且不要去修改自动生成的实例变量。(@property = getter + setter + _ ivar,这里的 _ ivar其实就是编译器帮我们生成的实例变量)

接下来你可以尝试去获取UILabel的实例变量列表:

[self ivarArray:[UILabel class]]

你会发现拿到的结果是这样的:

(
"_size",
"_highlightedColor",
"_numberOfLines",
"_measuredNumberOfLines",
"_baselineReferenceBounds",
"_lastLineBaseline",
"_previousBaselineOffsetFromBottom",
"_firstLineBaseline",
"_previousFirstLineBaseline",
"_minimumScaleFactor",
"_content",
"_synthesizedAttributedText",
"_defaultAttributes",
"_fallbackColorsForUserInterfaceStyle",
"_minimumFontSize",
"_lineSpacing",
"_layout",
"_scaledMetrics",
"_cachedIntrinsicContentSize",
"_contentsFormat",
"_cuiCatalog",
"_cuiStyleEffectConfiguration",
"_textLabelFlags",
"_adjustsFontForContentSizeCategory",
"__textColorFollowsTintColor",
"_preferredMaxLayoutWidth",
"_multilineContextWidth",
"__visualStyle"
)

但是跳转到UILabel.h,你会发现里面有好多的属性不包含在我们根据该方法得出的属性数组里面,而且使用该方法得到的属性在UILabel.h里面并没有。这个是什么原因呢?

先看一下好多UILabel里面的属性没有在数组里面打印问题:猜想应该是在UILabel.m里面使用了 @dynamic。导致没有自动生成getter、setter和ivar,所以没有在数组里面包含。

@synthsize:如果没有手动实现setter/getter方法那么会自动生成,自动生成_var变量。如果不写,默认生成getter/setter和_var。你也可以使用该关键字自己设置自动变量的名称。

@dynamic告诉编译器:属性的setter/getter需要用户自己实现,不自动生成,而且也不会产生_var变量。

也就是说在UILabel里面虽然有个text的属性,也许在UILabel.m里面已经包含:

@dynamic text;

这样的话在实现里面没有产生实例变量,只是手动实现了getter和setter,所以就不会显示text属性在刚才得到的数组里面了。

至于数组中有UILabel.h里面没有的变量,这个就好理解了,有可能在UILabel.m里面添加了一些实例变量或者在运行时添加了这些实例变量。

除此方法之外,你还可以使用class_copyPropertyList方法,这个是拿到的所有用 @property 声明的属性,包括在.m里面添加的属性(所以打印出来的可能要比真实在.h里面看到的多),具体实现和上面的获取方法类似:

- (NSArray *)propertyArr:(Class)cls {
unsigned count = 0;
objc_property_t *properties = class_copyPropertyList(cls, &count);
if (count == 0) {
return nil;
}
NSMutableArray *arr = [[NSMutableArray alloc] initWithCapacity:count];
for (int i = 0; i < count; i ++) {
objc_property_t property = properties[i];
NSString *propertyName = [NSString stringWithUTF8String:property_getName(property)] ;
[arr addObject:propertyName];
}
free(properties);
return arr;
}

其中的copyPropertyList方法解释如下:



记得使用过后也要调用free去释放数组。(PS:在源代码中暂未找到objc_property结构体的说明)因此,你可以通过使用该方法来实现字典或者json字符串转model操作:

+ (instancetype)zg_modelFromDic:(NSDictionary *)dataDic {
id model = [[self alloc] init];
unsigned int count = 0; objc_property_t *properties = class_copyPropertyList([self class], &count);
if (count == 0) {
return model;
}
for (int i = 0;i < count; i++) {
objc_property_t property = properties[i];
NSString *propertyName = [NSString stringWithUTF8String:property_getName(property)];
id value = dataDic[propertyName];
[model setValue:value forKey:propertyName];
}
free(properties);
return model;
}

两种方式均可实现model转换操作。

以上便是由class_copyIvarList所引发的思考。

转载请标明来源:http://www.cnblogs.com/zhanggui/p/8177400.html

class_copyIvarList方法获取实例变量问题引发的思考的更多相关文章

  1. Java——静态变量/方法与实例变量/方法的区别

    静态只能调用静态 非静态: 对象名.方法名 package ti; //通过两个类 StaticDemo.LX4_1 说明静态变量/方法与实例变量/方法的区别. class StaticDemo { ...

  2. Python staticmethod classmethod 普通方法 类变量 实例变量 cls self 概念与区别

    类变量 1.需要在一个类的各个对象间交互,即需要一个数据对象为整个类而非某个对象服务. 2.同时又力求不破坏类的封装性,即要求此成员隐藏在类的内部,对外不可见. 3.有独立的存储区,属于整个类.   ...

  3. System.getProperty()方法获取系统变量

    来自我的CSDN博客   今天在阅读JDBC的DriverManager类源码时,看到了这么一句代码: System.getProperty("jdbc.drivers"):    ...

  4. C#关于在返回值为Task方法中使用Thread.Sleep引发的思考

    起因 最近有个小伙伴提出了一个问题,就是在使用.net core的BackgroundService的时候,对应的ExecuteAsync方法里面写如下代码,会使程序一直卡在当前方法,不会继续执行,代 ...

  5. OC中实例变量可见度、setter、getter方法和自定义初始化方法

    在对类和对象有一定了解之后,我们进一步探讨实例变量的可见度等相关知识 实例变量的可见度分为三种情况:public(共有),protected(受保护的,默认),private(私有的),具体的不同和特 ...

  6. oc实例变量初始化方法

    1 使用实例setter方法 默认初始化方法 + setName:xxx setAge:xxx 2 使用实例功能类方法,默认初始化方法 + setName:xxx age:xxx3 使用实例初始化方法 ...

  7. JAVA类与对象(六)------实例变量与类变量的区别,实例方法和类方法的区别

    实例变量 实例变量声明在一个类中,但在方法.构造方法和语句块之外: 当一个对象被实例化之后,每个实例变量的值就跟着确定: 实例变量在对象创建的时候创建,在对象被销毁的时候销毁: 实例变量的值应该至少被 ...

  8. java入门---变量类型&类变量&局部变量&实例变量&静态变量

        在Java语言中,所有的变量在使用前必须声明.声明变量的基本格式如下:     type identifier [ = value][, identifier [= value] ...] ; ...

  9. JAVA类与对象---实例变量与类变量的区别,实例方法和类方法的区别

    实例变量 实例变量声明在一个类中,但在方法.构造方法和语句块之外: 当一个对象被实例化之后,每个实例变量的值就跟着确定: 实例变量在对象创建的时候创建,在对象被销毁的时候销毁: 实例变量的值应该至少被 ...

随机推荐

  1. 一些常用软件的静默安装参数(nsis,msi,InstallShield,Inno)

    打包的时候,经常需要安装一些其它的环境库,而又不想让用户繁锁的去选择,这时就需要静默安装,而不同的文件所加的参数了不一致,比如VS的环境库vcredist_x86.exe(这是32位的环境库)后面加/ ...

  2. 74、django之ajax补充

    之前的ajax使用都是依据jquery来使用的,本篇会先分析ajax的原生的js代码实现,还有jsonp的介绍,与OMR的一些遗漏补充. 本篇导航: js实现的ajax 同源策略与Jsonp 一.js ...

  3. Unity3D手机斗地主游戏开发实战(04)_出牌判断大小(已完结)

    之前我们实现了叫地主.玩家和电脑自动出牌主要功能,但是还有个问题,出牌的时候,没有有效性检查和比较牌力大小.比如说,出牌3,4,5,目前是可以出牌的,然后下家可以出任何牌如3,6,9. 问题1:出牌检 ...

  4. 实际应用中遇到TimedRotatingFileHandler不滚动的问题

    需求: 程序每天晚上8点和10点定时运行,期望日志按日期记录 添加Handler部分代码如下: formatter = logging.Formatter("%(asctime)s %(fi ...

  5. 【LintCode·容易】用栈模拟汉诺塔问题

    用栈模拟汉诺塔问题 描述 在经典的汉诺塔问题中,有 3 个塔和 N 个可用来堆砌成塔的不同大小的盘子.要求盘子必须按照从小到大的顺序从上往下堆 (如:任意一个盘子,其必须堆在比它大的盘子上面).同时, ...

  6. iOS 图片本地存储、本地获取、本地删除

    在iOS开发中.经常用到图片的本地化. iOS 图片本地存储.本地获取.本地删除,可以通过以下类方法实现. p.p1 { margin: 0.0px 0.0px 0.0px 0.0px; font: ...

  7. HDU3790-最短路径问题

    最短路径问题 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Total Sub ...

  8. linux上安装php7 memcache扩展 和 安装服务端memcached

    linux上安装memcached不算太困难.唯一让本人感到困难的是 php7的memcache扩展安装.真的蛋疼! 先说安装服务端 memcached 1. 首先安装Libevent事件触发管理器. ...

  9. Nginx负载均衡使用ip

    upstream test1{ server 192.168.1.213; server 192.168.1.37; } server { listen 80; # default backlog=2 ...

  10. 独家探寻阿里安全潘多拉实验室,完美越狱苹果iOS11.2.1

    知道如何从攻击的视角去发现漏洞,才能建立更安全的体系,促进了整个生态的良性发展.以阿里安全潘多拉实验室为例,在对移动系统安全研究的过程中,把研究过程中发现的问题上报给厂商,促进系统安全性的提升. 小编 ...