带大家学习Object-C Runtime锦集:property_getAttributes() 与 C++ typeid()

2017-01-14 10:45:10来源:http://www.jianshu.com/p/cefa1da5e775作者:月下伊人佳酒淳人点击

原文链接: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 编译器上得出的字符标识,不同的编译器的标识不一定相同喔。


参考资料:
苹果官方手册:
https://developer.apple.com/library/content/documentation/Cocoa/Conceptual/ObjCRuntimeGuide/Articles/ocrtPropertyIntrospection.html




最新文章

123

最新摄影

微信扫一扫

第七城市微信公众平台