天天看点

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 ** 通常是被假定为一个指向 nullable NSError 对象的 nullable 指针。

谢谢!!!