【老孟Flutter】强大的空安全

2021-04-22 15:41:13 浏览数 (1)

简介

空安全(Sound null safety)是 Dart 2.12 中新增的一项特性,空安全特性并不是 Dart 独有的,Kotlin, TypeScript, C#, Swift 等语言都有此特性,如果你了解这些语言的空安全特性及用法,那么下面关于 Dart 语言空安全特性的介绍你会感到非常熟悉,因为 Dart 语言空安全和其他语言基本一致。

版本要求

Dart 2.12和Flutter 2中提供了空安全性,对应到Flutter项目中,则需要在pubspec.yaml文件中添加如下配置:

代码语言:javascript复制
environment:
  sdk: ">=2.12.0 <3.0.0"

基本使用

变量

定一个 int 类型的变量,

代码语言:javascript复制
int age = null;

在没有空安全前,上面的代码是没有问题的,但当使用空安全后,在编译阶段出现异常,如下:

异常提示:null不能赋值给int变量。

这是空安全与以前最大的不同,默认情况下,变量不能为null(空安全以前任何类型都可以设置为null),更重要的是此异常在编译阶段即出现异常,无法编译通过。

如果想给一个变量赋值 null 要如何处理?只需在类型后面添加 ? 即可,如下:

代码语言:javascript复制
  int age = 1;
  int? ageNull = null;
  String? name = null;

类型后面跟操作符 ? 表示当前变量可为null。

变量的使用:

代码语言:javascript复制
String? name = null;
print('name length:${name?.length}');

非常简单,输出 name 字符串的长度,此时会发现,无法编译通过,异常如下:

修改如下:

代码语言:javascript复制
String? name = null;
print('name length:${name?.length}');

输出:

代码语言:javascript复制
flutter: name length:null

注意:上面 name 为 null,调用 name?.length 不会抛出异常,而是返回 null。

还可以有另外一种方式处理上面的异常:使用操作符 !

代码语言:javascript复制
String? name = null;
print('name length:${name!.length}');

上面的代码虽然可以编译通过,但运行时抛出异常,操作符 ! 表示检测当前变量不为 null,开发者需要保证变量不为 null,否则会抛出异常。

如果无法确认变量不为null,千万不要使用操作符 !

集合

看如下List集合:

代码语言:javascript复制
List<String> list;
List<String>? list1;
List<String?> list2;
List<String?>? list3;

他们的区别就是是否可为 null 的区别,List 表示 List 不为 null 而且集合中的 Item 也不能为 null。那么如下代码就是错误的:

代码语言:javascript复制
List<String> list;
//错误
list = null;
list.add(null);

List 集合说明如下:

类型

集合是否可为null

Item 是否可以为null

List

List?

List<String?>

List<String?>?

Map 类型也是同理,Map 中的 key 一般不为 null,下面的 Item 指的是Map 中的 value:

类型

集合是否可为null

Item 是否可以为null

Map<String,String>

Map<String,String>?

Map<String,String?>

Map<String,String?>?

方法参数
代码语言:javascript复制
void _incrementCounter(String? name) {
  print('name length:${name?.length}');
}

上面方法参数中加入了空安全,与变量用法一致。

class

定义一个类:

代码语言:javascript复制
class Person{
  
  final String name;

  Person(this.name);
}

有一个属性 name,属性类型为 String,说明此属性不能为 null,下面的使用是错误的:

代码语言:javascript复制
//错误,无法编译通过
var persion = Person(null);
//正确
var persion1 = Person('123');

将属性 name 改为可为 null:

代码语言:javascript复制
class Person{

  final String? name;

  Person(this.name);
}

那么下面的用法都是正确的:

代码语言:javascript复制
//正确
var persion = Person(null);
//正确
var persion1 = Person('123');
初始化 late

假设有一个属性,此属性的值来源于服务器或者其他方法,那么此时无法给此属性进行初始化,代码如下:

代码语言:javascript复制
String name;

此时会编译异常:

image-20210331172618734

提示我们必须要初始化,此情况使用关键字 late

代码语言:javascript复制
late String name;

使用此属性前 一定 要赋值,下面的用法运行时抛出异常:

代码语言:javascript复制
late String name;

void _incrementCounter() {
  print('name length:${name.length}');
}

异常:

正确用法:

代码语言:javascript复制
late String name;

void _incrementCounter() {
  name = '123';
  print('name length:${name.length}');
}

总结

空安全增加了2个操作符 ?! ,1个关键字 late

? :放在类型后面表示当前变量可为null,例如 int aint? b ,a 不能为null,而 b 可以。

! :放在变量后面,表示此变量值不为null,如果为null则会抛出异常,此操作符经常用于如下场景:一个方法的参数为非空类型(int),而传递给当前方法的变量是可为null的类型(int?),那么此时编译出现异常,在类型不变的情况下,在此变量的后面添加 ! ,表示当前变量不为null,代码如下:

代码语言:javascript复制
int? b = 2;

int _add(int a){
  return a 1;
}

//方法调用
_add(b!);

late:表示延迟初始化,通常用于延迟加载(比如网络请求),late 声明的变量在使用前一定要进行初始化。

0 人点赞