原文链接:https://abson.github.io/2016/10/31/typeid/
在逐渐开发 iOS 当中,不少人都听说过了 Runtime 运行时了,这个东西很多人都称它为一个黑魔法,它不仅拥有着神秘的效用,而且这也成为标识 iOS 领域的分水岭了,但是它是否真的很难呢?博主认为,很多事情,必须真正去尝试了,动手了去做了,等回过头来,也只是那么回事而已,所以别怕动手、思考。
在 Object-C 的 Runtime 当中,充当重要角色的property_getAttributes() 有着必不可少的地位 ,那什么是 property_getAttributes() 呢?
property_getAttributes: runtime 头文件中这个 C 语言方法了 ,其作用就是用来获取属性的真实类型。
但是近期研究C++中,发现了一个有趣的事情,就是C++中拥有的Rumtime,自然就发现了 typeid 这个有趣的事情了,什么是 typeid 呢?
typeid: 为RTTI提供的运算符,它允许程序像表达式提问:你的对象是什么类型
那么我们就会想, property_getAttributes() 究竟跟 typeid 是否存在什么联系呢?
答案是当然存在联系了,以下文章内容就围绕着这个主题展开。
首先我们看看 property_getAttributes() 的使用方式和其作用
以下是个例子:
首先创建一个 Object-C 的测试例子类
@interface RunTimeTestObj : NSObject
@property (nonatomic, assign) int num;
@end
@implementation RunTimeTestObj
@end
然后我们使用熟悉的 runtime 方法来获取这个测试类中的属性,并识别其类型。( 这些内容就是我们做 模型->字典 或 字典->模型 所用到的基础 )
unsigned int outCount = 0;
objc_property_t* propertys = class_copyPropertyList([RunTimeTestObj class], &outCount);
for (unsigned i = 0; i < outCount; i++) {
objc_property_t property = propertys[i];
assert(property != nil);
const char* name = property_getName(property);
NSLog(@"name: %s", name);
NSString* attrs = @(property_getAttributes(property));
NSLog(@"code: %@", attrs);
}
上面例子输出
name: num
attrs: Ti,N,V_num
上面输出我们可以看出,name 自然是这个属性的名称了,但是 attrs: Ti,N,V_num 这串字符串又是什么呢?
我来为大家解析一下:
通过阅读苹果官方文档可以看出
T: 在大写 T 后面是放的是该属性的数据类型
V: 在大写 V 后面放的是该属性的变量名称(因为我们知道 @property 实际上只是为我们编写好了 getter 和 setter 方法,并创建一个以下划线开头的变量)
输出中看出,T后面放的是 i,那么 i 又是什么类型呢?
然后带着这个问题,我找到了一张图
是不是过于复杂了,不怕,等下博主为你们简化一下,现在通过图我们可以理解为 i 类型, 就是 int 类型
但是 N 又代表什么呢?
然后带着这个问题,我找到了一张图!
由图片我们可以清晰的看到,N代表的是属性的非原子属性 nonatomic 的标识,当然我们加其他不同的标识也会有不同的字符,具体怎么样,请看图呀!
以上就是 property_getAttributes() 这个方法给到我们的信息了。
以下内容是 C++ 相同内容的对比了,没有兴趣的看官可以略过,直接看最后
那么C++ 中的typeid()能给到我们什么呢?我要说的是,runtime 技术都一样,OC 只是高级语言,typeid 实际上就是 property_getAttributes() 所使用的技术,我们再用实际例子看一看。
int p = 42;
double q = 43.f;
std::cout << typeid(p).name()<< " " << typeid(q).name() << "\n";
很显然,大家都知道结果输出: i d
这是跟 Object-C 中一样的套路呀,由此看出,runtime 技术都一样,理解好了,走到那里都不怕。
最后我为大家附上 OC 中总结 property_getAttributes() 得出的类型
NSString *const TypeInt = @"i"; 整数类型
NSString *const TypeShort = @"s"; 双字节类型
NSString *const TypeFloat = @"f"; float 浮点型
NSString *const TypeDouble = @"d"; double 双精度浮点型
NSString *const TypeLong = @"l"; long 长整型
NSString *const TypeLongLong = @"q";
NSString *const TypeChar = @"c"; char 单字节类型
NSString *const TypeBOOL1 = @"c"; bool 类型
NSString *const TypeBOOL2 = @"b"; bool 类型
NSString const TypePointer = @""; 指针类型
NSString *const TypeIvar = @"^{objc_ivar=}";
NSString *const TypeMethod = @"^{objc_method=}"; oc 方法类型
NSString *const TypeBlock = @"@?"; oc 闭包类型
NSString *const TypeClass = @"#"; oc class 类类型
NSString *const TypeSEL = @":"; oc slector 触发器方法类型
NSString *const TypeId = @"@"; oc 泛型id类型
最后说上一句,以上不管是 property_getAttributes() 还是 typeid() 得出的这些标识字符,只是在 XCode 的 LLVM 编译器上得出的字符标识,不同的编译器的标识不一定相同喔。