作者 | 弗拉德
起源 | 弗拉德(公众号: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 == 1a = 0;b = a++; // 在 b 赋值后将 a 减少 1。assert(a != b); // 1 != 0a = 0;b = --a; // 在 b 赋值前将 a 缩小 1。assert(a == b); // -1 == -1a = 0;b = a--; // 在 b 赋值后将 a 缩小 1。assert(a != b); // -1 != 0
关系运算符
关系运算符 | 形容 |
---|---|
== | 相等 |
!= | 不等于 |
> | 大于 |
< | 小于 |
>= | 大于等于 |
<= | 小于等于 |
要判断两个对象 x 和 y 是否示意雷同的事物应用 ==
即可。(在极少数状况下,可能须要应用 identical()
函数来确定两个对象是否完全相同)。
上面是 ==
运算符的一些规定:
- 假如有变量
x
和y
,且x
和y
至多有一个为null
,则当且仅当x
和y
均为null
时x == y
才会返回true
,否则只有一个为null
则返回false
。 x.==(y)
将会返回值,这里不论有没有y
,即y
是可选的。也就是说==
其实是x
中的一个办法,并且能够被重写。
上面的代码给出了每一种关系运算符的示例:
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
才是 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 + bvar a = 2; // 应用 = 赋值 (Assign using =)a *= 3; // 赋值并做乘法运算 Assign and multiply: a = a * 3assert(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语言——类中解说.