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指针。

谢谢!!!

你可能感兴趣的:(iOS __nullable 和 __nonnull 区别)