语法糖#02 Dart

2019-06-21

目录:

  1. ?:
  2. ??
  3. ?.
  4. ??=
  5. =>
  6. .. (Cascade notation)
  7. ~/
  8. Bonus: (Planned, not released) Spread operator: …

1. ?:

三目运算符,在很多编程语言中都有,这里做一个完整的描述。
当变量没有使用 if-else 的情况下有两种可能的值时,三目运算符提供了一种更为简洁的操作。
(1)普通写法:

int heightInCM = 185;
String heightCategory = "";
if(heightInCM > 175){
  heightCategory = "Tall";
}else{
  heightCategory = "short";
}

(2)使用三目运算符写法:

int heightInCM = 185;
String heightCategory = heightInCM > 175 ? "Tall" : "Short";

三目运算符的结构:

condition ? (statement if true) : (statement if false);

如果需要返回值,只需在前面加 return

return condition ? (statement if true) : (statement if false);

而不是像下面这样:

condition ? return (statement if true) : return (statement if false);

2. ??

空检查运算符,检查变量的值是否为空,如果为空则给它赋值。正如下面举的这个例子,如果person.namenull ,那么 name 会被赋值为 Emily CH。由此可以看出,空检查运算符极大的简化了我们对 null的处理。
(1)普通写法:

  String name = "";
  if(person.name == null){
    name = "Emily CH";
  }else{
    name = person.name;
  }
  
  OR
  
  String name = person.name != null ? person.name : "Emily CH";

(2)使用空检查运算符写法:

String name = person.name ?? "Emily CH";

3. ?.

假如有一个类 Point,有 xy 两个变量。

class Point {
  int x;
  int y;
  
  Point(this.x, this.y);
}

如果我们没有初始化 Point就取 xy 的值,那么它不是返回一个 null , 而是直接报错。

Point point;

print(point.x);

会抛出错误:

Unhandled exception:
NoSuchMethodError: The getter 'x' was called on null.
Receiver: null
Tried calling: x

如果相反,它只是返回一个null,那么我们可以轻松地处理并解决未初始化的问题。
普通写法:

  Point point;
  if(point != null){
    print(point.x);
  }else{
    print("No value");
  }

这样写也还行,但是它的限制是很明显的,举个栗子,我们解析 JSON 数据时,很多时候都是多层嵌套的。例如:

pages[0].contributors[0].authorDetails.basicInfo.firstName

在这个语句中,任何一个环节都有可能是 null 然后导致报错,例如 authorDetails 类中没有给 basicInfo这个字段赋值,那么 basicInfo 中取 firstName 的值就会报错。

如果要用上面的 if-else 去一个个环节检查,那将是一场噩梦。

?.有条件的成员访问运算符应运而生,这个运算符的作用可以简单粗略的说成是如果一个对象不为空,那么获取他的内部变量,否则返回 null
回到上面的第一个例子,我们用?. 运算符修改一下:

  Point point;
  int x = point ?. x ?? 0; 

这一段代码中,如果 Point 没有被初始化,这里不会报错而是返回一个 null ,然后 null 会被空检查运算符?? 捕捉并赋值 0 给 x
类似的,解析 JSON 数据的栗子我们也可以修改一下:

pages[0]?.contributors[0]?.authorDetails?.basicInfo?.firstName ?? "N/A";

现在,有任何一个环节为null ,将会返回 N/A

4. ??=

??= 简单的说就是如果左侧为空,则执行赋值。并且当变量为空时,只会分配一个值。这么说似乎有点难理解,看下面这个栗子,你应该就可以理解了。

  Point point;
  point ??= Point(x:1, y:2); //这个赋值会执行
  point ??= Point(x:3, y:4); //这个赋值不会执行

这样可以避免重复的赋值。

5. =>

=>Dart 中称为粗箭头运算符。它有两种应用场景。一是 相当于 {return x;} 的作用,二是作用于单条语句。例如:

  // 1. 相当于 `{return x;}` 的作用
  String demoFunction(String str){
    return str.toLowerCase();
  }

  String demoFunction(String str) => str.toLowerCase();

  // 2. 作用于单条语句
  void demoFunction(String str) => print(str);

6. ..

级联运算符更改对象属性(通常是在创建对象时)的一种简单方法,而不是获得对该对象的引用并逐个更改属性。举个粟子:
(1) 普通写法:

Point p = Point();
p.x = 3;
p.y = 6;

(2)使用级联运算符

    Point point = Point()
            ..x = 3
            ..y = 6;

当需要在Builder模式中设置许多属性时,这非常有用。例如:

    final addressBook = (AddressBookBuilder()
            ..name = 'emily'
            ..email = '[email protected]'
            ..phone = (PhoneNumberBuilder()
                    ..number = '1335668038'
                    ..label = 'home')
            .build())
        .build();

7. ~/

算术运算符,〜/运算符除以并返回结果的底数(整数部分)。举个粟子:

        int a = 3;
        int b = 7;
        int c = b ~/a;
        print(c); // 2

8. …

运算符作为集合的传播运算符,使用此运算符可以轻松解压缩列表。举个粟子:

  var demoList = [1,2,3,4,5];
  var otherList = [
          6,
          7,
          8,
          ...demoList,
          ]

9. 总结


文章是 Android 面向需求开发系列中的译文一篇,更多相关文章请关注。如若有什么问题,也可以通过扫描二维码发消息给我。转载请注明出处,谢谢!

2020-6-5

你可能感兴趣的:(语法糖#02 Dart)