Xcode6.3新特性之Nullability

Optional的定义

Optional是Objective-C没有的数据类型,是苹果引入到Swift语言中的全新类型,它可以有值,也可以没有值,当它没有值时,就是nil。此外,Swift的nil也和Objective-C有些不一样,在Objective-C中,只有对象才能为nil,而在Swift里,当基础类型(整形、浮点、布尔等)没有值时,也是nil,而不是一个初始值,没有初始值的值,是不能使用的,这就产生了Optional类型。Optional类型的值不能直接使用,需要拆包才可以取到值。

Nullability Annotations

Objective-C对象没有区分是optional还是non-optional。Swift编译器并不知道一个Objective-C对象是哪种类型,因此这种情况下编译器会隐式地将Objective-C的对象当成是non-optional。

XCode6.3引用Objective-C新特性解决这个问题:nullability annotations

The Core:_Nullable and _Nonnull

新特性的核心是增加两个新的注释:_Nullable_Nonnull。正如你所预料的, _Nullable表示对象可能是NULL或者nil,_Nonnull表示对象不能为空。 当你没有遵循规则时编译器就会报错。

@interface AAPLList : NSObject 
// ...
- (AAPLListItem * _Nullable)itemWithName:(NSString * _Nonnull)name;
@property (copy, readonly) NSArray * _Nonnull allItems;
// ...
@end

// --------------

[self.list itemWithName:nil]; // warning!

在任何使用C const 关键字的地方都可以使用_Nullable_Nonnull,但仅限于使用在指针类型上。然而在一般情况有更漂亮的方式书写这些注释:直接在圆括号后面使用不带下划线的nullablenonnull,只要它的类型是一个对象或者block指针。

- (nullable AAPLListItem *)itemWithName:(nonnull NSString *)name;
- (NSInteger)indexOfItem:(nonnull AAPLListItem *)item;

在声明属性时,也可以用同样的方式和修饰符写在一起。

@property (copy, nullable) NSString *name;
@property (copy, readonly, nonnull) NSArray *allItems;

不带下划线的形式更好看一些,但你仍然需要在你的头文件里设置每一个类型。为了让你更轻松,可以使用audited regions。

Audited Regions

为了更简单使用这些新注释,可以在头文件标记一定区域为nullability。在这个区域内,所有的指针类型会被假定为nonnull

NS_ASSUME_NONNULL_BEGIN
@interface AAPLList : NSObject 
// ...
- (nullable AAPLListItem *)itemWithName:(NSString *)name;
- (NSInteger)indexOfItem:(AAPLListItem *)item;

@property (copy, nullable) NSString *name;
@property (copy, readonly) NSArray *allItems;
// ...
@end
NS_ASSUME_NONNULL_END

// --------------

self.list.name = nil;   // okay

AAPLListItem *matchingItem = [self.list itemWithName:nil];  // warning!

为了安全要注意以下一条规则:

* typedef定义的类型要结合上下文看,不能假定它是nonnull
* 复杂的指针类型比如id *,一定要明确它的注释。例如一个指向nullable对象的non-nullable指针:_Nullable id * _Nonnull
* NSError **通常被假定指向nullable类型NSError的对象的nullable指针,用来通过方法参数返回错误。

Compatibility

* 编译过的代码可以正常运行,即使向它们传nil也不会报错。
* 现存代码在和Swift混编时,Swift编译器会对当前不安全的行为提出警告。
* nonnull不会影响性能。并且在运行时,你仍然可以检查被标记为nonnull的参数是否为nil。在向后兼容时这可能是必要的。

一般你可以把nullablenonnull简单的看成异常或断言一样,控制程序错误。特别当返回类型为non-nullable时不能返回nil,除非为了向后兼容。

在Xcode6.3发布了关键字__nullable__nonnull。由于和第三方库有潜在冲突,在Xcode7修改为_Nullable_Nonnull 。为了兼容Xcode6.3,要预先定义宏__nullable__nonnull来扩展同样的名字。

Back to Swift

现在添加Nullability Annotations到Objective-C的头文件中,在Swift代码中:

添加注释之前:

class AAPLList : NSObject, NSCoding, NSCopying { 
    // ...
    func itemWithName(name: String!) -> AAPLListItem!
    func indexOfItem(item: AAPLListItem!) -> Int

    @NSCopying var name: String! { get set }
    @NSCopying var allItems: [AnyObject]! { get }
    // ...
}

之后:

class AAPLList : NSObject, NSCoding, NSCopying { 
    // ...
    func itemWithName(name: String) -> AAPLListItem?
    func indexOfItem(item: AAPLListItem) -> Int

    @NSCopying var name: String? { get set }
    @NSCopying var allItems: [AnyObject] { get }
    // ...
}

Swift代码现在变得更加简洁。这只是一个微小的改变,但在使用时会变得更加方便。


哪里有问题提出修改意见~

你可能感兴趣的:(Xcode6.3新特性之Nullability)