OC数据类型隐式转换
NSObject :
所有的OC类型的root类是NSObject,所以任何直接或继承链包含NSObject的OC实例对象都可以隐式转换为NSObject实例对象指针。
id:
Id是objc_object指针类型,官方注释为指向任何类的实例的一个指针。注意这里并不是指id是一个NSObject类型的实例,之所以能够指向NSObject实例对象是NSObject也遵循objc_object对应的结构,因此任何遵循objc_object结构的实例都可隐式转换成id类型,包括Class
Class:
Class官方注释为一个OC类的隐式类型,雷同 id,可以看见objc_class 是继承自 objc_object的所以Class可以隐式转换为id类型
protocol:
protocol 实际上是一个id类型,所以我们再声明协议类型的指针的时候通常都是 id … delegate 也就是说protocol是可以隐式转换为id类型的
根据可以隐式转换为id类型的特性,任何遵循相同协议的类型都可以相互之间进行隐式转换。
关于隐式类型转换使用的问题:
根据以上说明隐式转换使用非常灵活尤其id 类型更不安全。所以我们要做好运行时判断防止不在预期的错误。
同时根据以上说明子类类型是可以隐式转换为继承链的类型并且可以安全如预期的执行。
基本数据类型的隐式转换
double、int、long、float、unsigned
基本数据类型的转换遵循:当做运算时例如加减乘除都会转换成精度最高的类型,当进行赋值是都会被转换成目标精度的类型,需要注意的是无符号运算中的负数问题,例如
NSUInteger uint = 0
NSInteger duInt = -1
If duInt < uint {
print(“duInt小于 uint”)
}else {
print(“duInt 大于 uint”)
}
以上打印duInt大于uint。这不符合常理,这是因为在运算时编译器将 -1 转换为 NSUInteger 的最大值 -3.618904e+15 18446744073709551615, 如果是 -2 则为 18446744073709551614以此类推。当然也好解决,需要我们显示的把无符号转换一下就可以正常运算了
If duInt < (NSInteger)uint {
}
我们也应该尽量避免 NSUInteger 的比较操作(实际上并不能避免,OC的.count 属性都为NSUInteger)我们也只能在做比较时关注另一个参数的正确性了。
结论:
隐式转换是ide自发的行为,所以正确的隐式转换是安全的。可以总结为集合的转换,大的集合转换成小的集合是安全的,例如子类转换成父类,而基本类型小的精度转换成的大精度,例如int、double,这也是为了保证运算准确。同时不合适的隐式转换ide也会适当的给出warning例如用父类的指针类型指向子类的实例。
但是ide只能保证逻辑上的安全,如果业务上还有要求需要根据业务需求来做,例如需要显示一个整数但是后台返回了浮点数我们运算之后便可能出现显示问题。
说到业务需求在实际工作中有时我们并不适合去使用隐式转换,而更适合使用显示强制转换,这无疑是有很大风险的,这就需要我们代码的控制。