美文网首页Flutter
Dart 2 (四) 运算符和表达式

Dart 2 (四) 运算符和表达式

作者: pstommy | 来源:发表于2018-09-13 10:46 被阅读0次

    运算符和表达式

    描述 运算符
    一元后缀 expr++ expr-- () [] . ?.
    一元前缀 -expr !expr ~expr ++expr --expr
    乘除 * / % ~/
    加减 + -
    位移 << >>
    按位与 &
    按位或
    按位异或 ^
    关系和类型测试 >= > <= < as is is!
    等于 == !=
    逻辑与 &&
    逻辑或
    如果空 ??
    条件 expr1 ? expr2 : expr3
    级联 ..
    赋值 = *= /= ~/= %= += -= <<= >>= &= ^= ??=

    注:按位或是“|” ,赋值 还有“|=”,暂时没有找到“|”的转义字符,以后补上;

    当您使用运算符时,您将创建表达式。下面是一些运算符表达式的例子:

    a++
    a + b
    a = b
    a == b
    c ? a : b
    a is T
    

    在运算符表中,每个运算符的优先级都高于后面行中的运算符。例如,模运算符%的优先级高于相等运算符==(因此在此之前执行),后者的优先级高于逻辑和运算符&&。优先级意味着以下两行代码以相同的方式执行:

    // Parentheses improve readability.
    if ((n % i == 0) && (d % i == 0)) ...
    
    // Harder to read, but equivalent.
    if (n % i == 0 && d % i == 0) ...
    

    算术运算符

    运算符 含义
    +
    -
    -expr 一元减号
    *
    /
    ~/ 除,返回整数结果
    % 取模

    例子:

    assert(2 + 3 == 5);
    assert(2 - 3 == -1);
    assert(2 * 3 == 6);
    assert(5 / 2 == 2.5); // Result is a double
    assert(5 ~/ 2 == 2); // Result is an int
    assert(5 % 2 == 1); // Remainder
    
    assert('5/2 = ${5 ~/ 2} r ${5 % 2}' == '5/2 = 2 r 1');
    

    Dart还支持前缀和后缀递增和递减运算符。

    var a, b;
    
    a = 0;
    b = ++a; // Increment a before b gets its value.
    assert(a == b); // 1 == 1
    
    a = 0;
    b = a++; // Increment a AFTER b gets its value.
    assert(a != b); // 1 != 0
    
    a = 0;
    b = --a; // Decrement a before b gets its value.
    assert(a == b); // -1 == -1
    
    a = 0;
    b = a--; // Decrement a AFTER b gets its value.
    assert(a != b); // -1 != 0
    

    等式和关系运算符

    下表列出了等式和关系运算符的含义。

    运算符 含义
    == 等于
    != 不等于
    > 大于
    < 小于
    >= 大与等于
    <= 小于等于

    要测试两个对象x和y是否表示同一事物,使用==运算符。(在您需要知道两个对象是否完全相同的情况下,可以使用 identical()函数。)下面是==操作符的工作方式:

    如果x或y为空,如果两者都为空返回true,如果只有一个为空返回false。

    返回方法调用的结果x.==(y)。(对,==这样的操作符是在其第一个操作数上调用的方法。您甚至可以覆盖许多操作符,包括==,您将在可覆盖操作符中看到。

    下面是使用等式运算符和关系运算符的示例:

    assert(2 == 2);
    assert(2 != 3);
    assert(3 > 2);
    assert(2 < 3);
    assert(3 >= 3);
    assert(2 <= 3);
    

    类型检测操作符

    as, is, is!在运行时,操作符可以方便地检查类型。

    运算符 含义
    as 定型
    is 如果对象具有指定的类型,则为True
    is! 如果对象具有指定的类型,则为False

    如果obj 实现了 T 指定的接口 ,obj is T的结果为真,例如 obj is Object 总是为真。

    使用as操作符将对象转换为特定类型。通常,您应该使用它作为在对象上进行is测试的简写,后面是使用该对象的表达式。例如,考虑以下代码:

    if (emp is Person) {
      // Type check
      emp.firstName = 'Bob';
    }
    

    您可以使用as运算符使代码更短:

    (emp as Person).firstName = 'Bob';
    

    注意:以上代码不相等。如果emp为空或不为Person,第一个示例(带is)什么也不做;第二个(带有as)抛出异常。

    赋值操作符

    正如您已经看到的,可以使用=操作符赋值。如果指定的变量为空,则使用??=运算符。

    // 赋值给a
    a = value;
    // 如果b为空,则赋值给b;否则,b保持不变
    b ??= value;
    

    复合赋值运算符(如+=)将操作与赋值组合在一起。
    = -= /= %= >>= ^=
    += *= ~/= <<= &= |=

    下面的示例使用赋值运算符和复合赋值运算符:

    var a = 2; // 使用=赋值
    a *= 3; // 相乘后赋值: a = a * 3
    assert(a == 6);
    

    逻辑运算符

    您可以使用逻辑运算符反转或组合布尔表达式。

    运算符 含义
    !expr 反转表达式(将false改为true,反之亦然)
    - 逻辑或
    && 逻辑与

    逻辑或为“||”,暂时没有找到转义字符,表格未完善!

    下面是一个使用逻辑运算符的例子:

    if (!done && (col == 0 || col == 3)) {
      // ...Do something...
    }
    

    按位和移位操作符

    运算符 含义
    & 按位与
    ^ 异或运算符
    ~expr 一元位补码(0变成1;1变成0)
    << 左移
    >> 右移

    “|”按位或运算符

    这里有一个使用位和移位运算符的例子:

    final value = 0x22;
    final bitmask = 0x0f;
    
    assert((value & bitmask) == 0x02); // AND
    assert((value & ~bitmask) == 0x20); // AND NOT
    assert((value | bitmask) == 0x2f); // OR
    assert((value ^ bitmask) == 0x2d); // XOR
    assert((value << 4) == 0x220); // Shift left
    assert((value >> 4) == 0x02); // Shift right
    

    条件表达式

    Dart有两个运算符,可以让你简洁地计算可能需要if-else语句的表达式:
    condition ? expr1 : expr2
    如果条件为真,计算expr1(并返回其值);否则,计算并返回expr2的值。
    expr1 ?? expr2
    如果expr1非空,返回其值;否则,计算并返回expr2的值。

    当需要基于布尔表达式分配值时,考虑使用?:。

    var visibility = isPublic ? 'public' : 'private';
    

    如果布尔表达式检测为空,请考虑使用??。

    String playerName(String name) => name ?? 'Guest';
    

    下面的示例用另外两种方式编写,但不那么简洁:

    // Slightly longer version uses ?: operator.
    String playerName(String name) => name != null ? name : 'Guest';
    
    // Very long version uses if-else statement.
    String playerName(String name) {
      if (name != null) {
        return name;
      } else {
        return 'Guest';
      }
    }
    

    级联符号(..)

    级联(..)允许对同一对象进行一系列操作。除了函数调用外,您还可以访问同一对象上的字段。这通常可以省去创建临时变量的步骤,并允许您编写更流畅的代码。
    考虑以下代码:

    querySelector('#confirm') // Get an object.
      ..text = 'Confirm' // Use its members.
      ..classes.add('important')
      ..onClick.listen((e) => window.alert('Confirmed!'));
    

    第一个方法调用querySelector()返回选择器对象。级联表示法后面的代码操作这个选择器对象,忽略可能返回的任何后续值。
    上例等价于:

    var button = querySelector('#confirm');
    button.text = 'Confirm';
    button.classes.add('important');
    button.onClick.listen((e) => window.alert('Confirmed!'));
    

    你也可以嵌套你的级联。例如:

    final addressBook = (AddressBookBuilder()
          ..name = 'jenny'
          ..email = 'jenny@example.com'
          ..phone = (PhoneNumberBuilder()
                ..number = '415-555-0100'
                ..label = 'home')
              .build())
        .build();
    

    要小心在返回实际对象的函数上构造级联。例如,以下代码失败:

    var sb = StringBuffer();
    sb.write('foo')
      ..write('bar'); // Error: method 'write' isn't defined for 'void'.
    

    调用sb.write()返回void,并且不能在void上构造级联。
    注意:严格地说,级联的“双点”符号不是运算符。这只是Dart语法的一部分。

    其他运算符

    您已经在其他示例中看到了大多数剩余的操作符:

    运算符 名称 含义
    () 函数 表示函数调用
    [] list 指List中指定索引处的值
    . 成员访问 指表达式的属性;例子:foo. 从表达式foo中选择属性栏
    ?. 有条件的成员访问 类似 .,但最左边的操作数可以为空;例: foo?.bar 从表达式foo中选择属性bar,除非foo为空(在这种情况下foo的值是?.bar为空)

    相关文章

      网友评论

        本文标题:Dart 2 (四) 运算符和表达式

        本文链接:https://www.haomeiwen.com/subject/orrbgftx.html