我们都知道在
Swift
中,可以使用!
和?
来表示一个对象是optional
的还是non-optional
,如view?
和view!
。而在Objective-C
中则没有这一区分,view
既可表示这个对象是optional
,也可表示是non-optional
。这样就会造成一个问题:在Swift
与Objective-C
混编时,Swift
编译器并不知道一个Objective-C
对象到底是optional
还是non-optional
,因此这种情况下编译器会隐式地将Objective-C
的对象当成是non-optional
。
问题解决
苹果在Xcode 6.3
引入了一个Objective-C
的新特性:nullability annotations
。这一新特性的核心是两个新的类型注释: __nullable
和 __nonnull
。从字面上我们可以猜到,__nullable
表示对象可以是NULL
或nil
,而__nonnull
表示对象不应该为空。当我们不遵循这一规则时,编译器就会给出警告。
如果需要每个属性或每个方法都去指定nonnull
和nullable
,是一件非常繁琐的事。苹果为了减轻我们的工作量,专门提供了两个宏:NS_ASSUME_NONNULL_BEGIN, NS_ASSUME_NONNULL_END。在这两个宏之间的代码,所有简单指针对象都被假定为nonnull
,因此我们只需要去指定那些nullable
的指针。
NS_ASSUME_NONNULL_BEGIN
@interface TestObject : NSObject
@property (nonatomic, strong) NSString *testString;
@property (nonatomic) BOOL isRight;
@end
NS_ASSUME_NONNULL_END
使用规则
typedef
定义的类型的nullability
特性通常依赖于上下文,即使是在Audited
-Regions
中,也不能假定它为nonnull
。- 复杂的指针类型(如
id
*)必须显示去指定是non
null
还是nullable
。例如,指定一个指向nullable
对象的nonnulla
指针,可以使用”__nullable id
*__nonnull
”。 - 我们经常使用的
NSError
**通常是被假定为一个指向nullable NSError
对象的nullable
指针。