29-dart-null-effective

29. null使用最佳实践

简介

null可能是大家在编写程序中最为头疼的一个东西,稍不留意的情况下就有可能使用到了这个空字符。所以dart在2.12引入了nll safety,默认情况下强制所有的类型都不为null,只有在你认为它可以为null的时候才可以设置为null。

虽然有了null safety,但是这里还有一些我们需要考虑的null的最佳实践。

不需要初始化对象为null

在dart2.12之后,所有的对象都强制为非空的,除非你显示指定其为可空的对象。

如果一个对象可以为空,那么我们可以这样指定:

String? name;

如果定义一个对象可以为空,那么对dart来说会隐式对其初始化为null。

所以下面的显示初始化为null是完全没有必要的:

String? name=null;

同样的,如果参数是一个可以为空的对象,那么dart也会将其初始化为null,我们也没有必要显示去设置其值:

void echoName(String? name){
    print(name);
}

null的三元操作符

所谓三元就是有三个变量,我们常见的三元操作符就是?:,通常来说是这样用的:

name==null?true:false;

上面的逻辑实际上是把一个null转换成了一个bool类型。

为了实现这个功能,dart提供了一个更加简洁的操作符??, 可以这样使用:

name??false;

上面的代码表示如果name是空,则返回false。

注意,这里只是返回值改变了,但是name值本身并没有变化,也不会将name从一个可为空的类型,变成不为空的类型。所以如果我们在if语句里面对字符进行判断,则还是需要显示进行null的比较:

int measureMessage(String? message) {
  if (message != null && message.isNotEmpty) {
      // dart知道message不为空
    return message.length;
  }

  return 0;
}

如果这样编写,则会出现异常:

int measureMessage(String? message) {
  if (message?.isNotEmpty ?? false) {
    //dart并不知道message不为空
    return message!.length;
  }

  return 0;
}

如果在使用中需要判断类型是否为空,则不要使用late

late是做什么用的呢?late表示该类型目前不会初始化,但是会在未来的某个时间对其进行初始化。

所以,如果你用late表示某个类型,那么在后续使用的时候是不需要进行手动判断该类型是否为空的。

如果你仍然要手动判断,那么就没必要设置该类型为late。

本地变量的类型提升

dart有一个非常好的特性,就是当我们判断一个变量不为空之后,该变量就会被提升为非空变量。

当提升为非空变量之后,就可以自由访问该非空变量内部的属性和方法了。

但是可惜的是,dart中的类型提升只是针对与local变量或者参数而言的,对于类变量或者是top level的变量并不适用,所以我们需要将这些变量拷贝到本地变量,从而使用类型提升的特性。

我们看下面的例子:

class UploadException {
  final Response? response;

  UploadException([this.response]);

  @override
  String toString() {
    var response = this.response;
    if (response != null) {
      return 'Could not complete upload to ${response.url} '
          '(error code ${response.errorCode}): ${response.reason}.';
    }

    return 'Could not upload (no response).';
  }
}

其中UploadException中的response是一个顶级变量,虽然我们对其进行测试是否为空,但是在使用的过程中还是不能直接访问其内部的属性,因为response可能为空。

为了使用dart的类型提升的特性,我们可以将顶级的变量赋值给一个本地变量,从而在null测试之后,自动将其提升为非空的类型,从而直接访问其内部的属性。

总结

以上就是dart中null用法的最佳实践。

最后更新于