乐趣区

关于flutter:Flutter-110Flutter手把手教程Dart语言运算符

作者 | 弗拉德
起源 | 弗拉德(公众号:fulade_me)

运算符

运算符是一种通知编译器执行特定的数学或逻辑操作的符号。Dart 语言内置了丰盛的运算符,并提供了以下类型的运算符:算术运算符、关系运算符、类型判断运算符、赋值运算符、逻辑运算符、按位和移位运算符、条件表达式、级联运算符以及其余运算符

算数运算符

算数运算符 形容
+
– 表达式 一元负, 也能够作为反转(反转表达式的符号)
*
/
~/ 除并取整
% 取模

示例:

assert(2 + 3 == 5);
assert(2 - 3 == -1);
assert(2 * 3 == 6);
assert(5 / 2 == 2.5); // 后果是一个浮点数
assert(5 ~/ 2 == 2); // 后果是一个整数
assert(5 % 2 == 1); // 取余

assert('5/2 = ${5 ~/ 2} r ${5 % 2}' == '5/2 = 2 r 1');

Dart 还反对自增自减运算符:

自增自减运算符 形容
++ var var = var + 1 (表达式的值为 var + 1)
var ++ var = var + 1 (表达式的值为 var)
— var var = var – 1 (表达式的值为 var – 1)
var — var = var – 1 (表达式的值为 var)

示例:

var a, b;

a = 0;
b = ++a; // 在 b 赋值前将 a 减少 1。assert(a == b); // 1 == 1

a = 0;
b = a++; // 在 b 赋值后将 a 减少 1。assert(a != b); // 1 != 0

a = 0;
b = --a; // 在 b 赋值前将 a 缩小 1。assert(a == b); // -1 == -1

a = 0;
b = a--; // 在 b 赋值后将 a 缩小 1。assert(a != b); // -1 != 0

关系运算符

关系运算符 形容
== 相等
!= 不等于
> 大于
< 小于
>= 大于等于
<= 小于等于

要判断两个对象 x 和 y 是否示意雷同的事物应用 == 即可。(在极少数状况下,可能须要应用 identical() 函数来确定两个对象是否完全相同)。
上面是 == 运算符的一些规定:

  1. 假如有变量 xy,且 xy 至多有一个为 null,则当且仅当 xy 均为 nullx == y 才会返回 true,否则只有一个为 null 则返回 false
  2. x.==(y) 将会返回值,这里不论有没有y,即 y 是可选的。也就是说 == 其实是 x 中的一个办法,并且能够被重写。

上面的代码给出了每一种关系运算符的示例:

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

类型判断运算符

asisis! 运算符是在运行时判断对象类型的运算符。

类型判断运算符 形容
as 类型转换(也用作指定类前缀))
is 如果对象是指定类型则返回 true
is! 如果对象是指定类型则返回 false

当且仅当 obj 对象 实现了 T 的接口,obj 对象 is T 才是 true。例如 obj 对象 is Object 总为 true,因为所有类都是 Object 的子类。

仅当你确定这个对象是该类型的时候,你才能够应用 as 操作符能够把对象转换为特定的类型。例如:

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

如果你不确定这个对象类型是不是 T,请在转型前应用 is T 查看类型。

if (emp is Person) {
  // 类型查看
  emp.firstName = 'Bob';
}

你能够应用 as 运算符进行缩写:

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

赋值运算符

能够应用 = 来赋值,同时也能够应用 ??= 来为值为 null 的变量赋值。

// 将 value 赋值给 a (Assign value to a)
a = value;
// 当且仅当 b 为 null 时才赋值
b ??= value;

+= 这样的赋值运算符将算数运算符和赋值运算符组合在了一起。

= –= /= %= >>= ^=
+= *= ~/= <<= &= =

上面的例子展现了如何应用赋值以及复合赋值运算符:

a += b      // 就 等同于 a = a + b
var a = 2; // 应用 = 赋值 (Assign using =)
a *= 3; // 赋值并做乘法运算 Assign and multiply: a = a * 3
assert(a == 6);

逻辑运算符

类型判断运算符 形容
! 表达式 对表达式后果取反(行将 true 变为 false,false 变为 true)
|| 逻辑或
&& 逻辑与

上面是应用逻辑表达式的示例:

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

按位和移位运算符

按位和移位运算符 形容
& 按位与
| 按位或
^ 按位异或
~ 表达式 按位取反(行将“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)

条件表达式

条件 ? 表达式 1 : 表达式 2:如果条件为 true,执行 表达式 1并返回执行后果,否则执行 表达式 2 并返回执行后果。
表达式 1 ?? 表达式 2:如果 表达式 1 为非 null 则返回其值,否则执行 表达式 2 并返回其值。
如果赋值是依据布尔表达式则思考应用 ?:

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

如果赋值是依据断定是否为 null 则思考应用 ??

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

上述示例还能够写成至多上面两种不同的模式,只是不够简洁:

// 绝对应用 ?: 运算符来说略微长了点。(Slightly longer version uses ?: operator).
String playerName(String name) => name != null ? name : 'Guest';

// 如果应用 if-else 则更长。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 返回了一个 Selector 对象,前面的级联操作符都是调用这个 Selector 对象的成员并疏忽每个操作的返回值。

下面的代码相当于:

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'); // 出错:void 对象中没有办法 write (Error: method 'write' isn't defined for'void').

上述代码中的 sb.write() 办法返回的是 void,返回值为 void 的办法则不能应用级联运算符。

其余运算符

大多数其它的运算符,曾经在其它的示例中应用过:

运算符 名字 形容
() 应用办法 代表调用一个办法
[] 拜访 List 拜访 List 中特定地位的元素
. 拜访成员 成员拜访符
?. 条件拜访成员 与上述成员拜访符相似,然而右边的操作对象不能为 null,例如 foo?.bar,如果 foo 为 null 则返回 null,否则返回 bar

更多对于 ., ?... 运算符介绍,会在下一章 Flutter 手把手教程 Dart 语言——类 中解说.

退出移动版