iOS __nullable 和 __nonnull 区别
//联系人:石虎 QQ:1224614774 昵称:嗡嘛呢叭咪哄
一、概念
两个新的类型修饰:__nullable和__nonnull。从字面上我们可知,__nullable表示对象可以是NULL 或 nil,而__nonnull表示对象不应该为空。
当我们不遵循这一规则时,编译器就会给出警告。在 Xcode 7 中,为了避免与第三方库潜在的冲突,苹果把__nonnull/__nullable改成_Nonnull/_Nullable。再加上苹果同样支持了没有下划线的写法nonnull/nullable,于是就造成现在有三种写法这样混乱的局面。
但是这三种写法本质上都是互通的,只是放的位置不同,
二、声明属性
方法返回值修饰:
- (nullableNSString*)method;
- (NSString* __nullable)method;
- (NSString* _Nullable)method;
声明属性的修饰:
@property(nonatomic,copy,nullable)NSString*aString;
@property(nonatomic,copy)NSString* __nullableaString;
@property(nonatomic,copy)NSString* _Nullable aString;
方法参数修饰:
- (void)methodWithString:(nullableNSString*)aString;
- (void)methodWithString:(NSString* _Nullable)aString;
- (void)methodWithString:(NSString* __nullable)aString;
三、总结
而对于 双指针类型对象 、 Block 的返回值 、 Block 的参数 等,这时候就不能用nonnull/nullable修饰,只能用带下划线的__nonnull/__nullable或者_Nonnull/_Nullable:
aString 属性默认是nonnull的,methodWithString:方法的返回值也是nonnull,而方法的参数 str 被显式指定为nullable。
苹果还制定了以下几条规则:
通过typedef定义的类型的 nullability 特性通常依赖于上下文,即使是在 Audited Regions 中,也不能假定它为nonnull;
对于复杂的指针类型(如id * )必须显式去指定是nonnull还是nullable。例如,指定一个指向nullable对象的nonnull指针,可以使用__nullable id * __nonnull;
我们经常使用的 NSError ** 通常是被假定为一个指向nullableNSError 对象的nullable指针。