小知识点iOS开发好东西

iOS关键字:nullable,nonnull,null_res

2017-02-06  本文已影响2598人  江小凡

当我们在查看苹果iOS各种开发框架的头文件时,在属性和方法的参数,方法的返回值的修饰中·经常会看到nullablenonnullnull_resettablenull_unspecified等关键字,比如:

那么这些关键字是什么意思呢?有什么作用呢?下面让我们一起仔细研究研究。

其实这几个关键字是苹果在Xcode 6.3引入的一个Objective-C的新特性:nullability annotations。在Swift中可以使用!?来表示一个对象是optional的还是non-optional,如button?button!。而在Objective-C中则没有这一区分,button即可表示这个对象是optional,也可表示是non-optioanl。这样就会造成一个问题:在SwiftObjective-C混编时,Swift编译器并不知道一个Objective-C对象到底是optional还是non-optional,因此这种情况下编译器会隐式地将Objective-C的对象当成是non-optional。引入nullability annotations一方面为了让iOS程序员平滑地从Objective-C过渡到Swift,另一方面也促使开发者在编写Objective-C代码时更加规范,减少同事之间的沟通成本。

下面让我们来看看每个关键字都是如何使用的吧!

在属性中使用nullable和nonnull:

nullable: 表示修饰的属性或参数可以为空,书写方式有三种:
方式一:

@property(nonatomic,strong,nullable) NSString * name;

方式二:

@property(nonatomic,strong) NSString *_Nullable name;

方式三:

@property(nonatomic,strong) NSString *__nullable name;

nonnull:非空,表示修饰的属性或参数不能为空,书写方式有三种:
方式一:

@property(nonatomic,strong,nonnull) NSNumber * price;

方式二:

@property(nonatomic,strong) NSNumber *_Nonnull price;

方式三:

@property(nonatomic,strong) NSNumber *__nonnull price;

注意:nonnull,nullable只能修饰对象,不能修饰基本数据类型

@property(nonatomic,assign,nonnull) int age;


在方法中使用nullable和nonnull:

nullable:
方式一:

 - (nullable NSString *)buyBook:(nullable NSString *)book;

方式二:

 - (NSString *__nullable)buyBook:( NSString *__nullable)book;

方式三:

 - (NSString *_Nullable)buyBook:( NSString *_Nullable)book;

在方法中使用nonnull与nullable方式相同,就不在此赘述了。

Nonnull区域设置(Audited Regions)
在查看苹果iOS各种开发框架的头文件时,我们发现都是用nullable修饰的属性和方法,为什么都没看到nonnull修饰的属性和方法呢?
事实上,如果需要每个属性或每个方法的参数和返回值都去指定nonnull和nullable,是一件非常繁琐的事。苹果为了减轻我们的工作量,专门提供了两个宏:NS_ASSUME_NONNULL_BEGIN和NS_ASSUME_NONNULL_END。苹果用NS_ASSUME_NONNULL_BEGIN,NS_ASSUME_NONNULL_END这两个宏来统一给属性和方法参数和返回值加上nonnull修饰,NS_ASSUME_NONNULL_BEGIN 和 NS_ASSUME_NONNULL_END之间,定义的所有对象属性和方法默认都是nonnull。如UIButton的头文件,下图所示:



用法示例:

 NS_ASSUME_NONNULL_BEGIN

 - (NSString *)buyPen:( NSString *)book;
 
 NS_ASSUME_NONNULL_END

 NS_ASSUME_NONNULL_BEGIN
 
 @property(nonatomic,strong) NSNumber * age;
 
 NS_ASSUME_NONNULL_END

不过,为了安全起见,苹果还制定了几条规则:

1.typedef定义的类型的nullability特性通常依赖于上下文,即使是在Audited Regions中,也不能假定它为nonnull
2.复杂的指针类型(如id *)必须显示去指定是nonnull还是nullable。例如,指定一个指向nullable对象的nonnull指针,可以使用”__nullable id * __nonnull”
3.我们经常使用的NSError **通常是被假定为一个指向nullable NSError对象的nullable指针。

null_resettable的使用:

** null_resettable: get方法:不能返回为空,set方法可以为空**
只有一种书写方式:

 @property(nonatomic,strong,null_resettable) NSNumber * number;

当属性策略中使用了null_resettable修饰,就必须保证该属性的get方法返回不为空,否则编译器会如上图那样报警告⚠️。可以在set方法或get方法中做非空处理,以下是在get方法中做处理:

- (NSNumber *)number{
    
    if (_number == nil) {
        _number = @11;
    }
    return _number;
}

null_unspecified的使用:

null_unspecified:不确定是否为空,使用方式有三种:

方式一:

 @property(nonatomic,strong) NSNumber *_Null_unspecified height;

方式二:

 @property(nonatomic,strong) NSNumber *__null_unspecified height;

方式三:

 @property(nonatomic,strong,null_unspecified) NSNumber * height;

至此,nullable,nonnull,null_resettable,null_unspecified这四个关键字的意思以及用法就介绍完了,希望能帮到大家。

上一篇下一篇

猜你喜欢

热点阅读