dart定义了下表所示的运算符。你可以重写许多这些运算符。

描述 运算符
一元后缀 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

在之前的操作符表中,操作符的优先级由其所在行定义,上面行内的操作符优先级大于下面行内的操作符。例如,乘法类型操作符%的优先级比等价操作符==要高,而==操作符的优先级又比逻辑与操作符&&要高。这些操作符的优先级顺序将在下面的两行代码中体现出来:

// 1.使用括号来提高可读性
if ((n % i == 0) && (d % i == 0)) // 2.难以阅读,但是和上面等价
if (n % i == 0 && d % i == 0)

警告:对于二元运算符,其左边的操作数将会决定使用的操作符的种类。例如,当你使用一个 Vector 对象以及一个 Point 对象时, aVector + aPoint 使用的 + 是由Vector 所定义的。


算术运算符

dart支持常用的算术运算符,如下表所示。

操作符 含义
+
-
-expr 一元减号,也被命名为负号(使后面表达式的值反过来)
*
/
~/ 返回一个整数值的除法
% 取余,除法剩下的余数

示例:

assert(2 + 3 == 5);
assert(2 - 3 == -1);
assert(2 * 3 == 6);
assert(5 / 2 == 2.5); // 结果是double类型
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
assert(a == b); // 1 == 1 a = 0;
b = a++; //在b获得值后自增a
assert(a != b); // 1 != 0 a = 0;
b = --a; // 在b获得其值之前自减a
assert(a == b); // -1 == -1 a = 0;
b = a--; // 在b获得值后自减a
assert(a != b); // -1 != 0

等式和关系运算符

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

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

要测试两个对象x和y是相等,请使用= =运算符。在极少数情况下,您需要知道两个对象是否是完全相同的对象,请改用experation()函数。以下是==运算符的工作原理:

  1. 如果x或y为空,如果两者都为空,则返回true;如果只有一个为空,则返回false。
  2. 返回一个函数调用的结果:x.==(y)。(这个调用是正确的,像==这样的运算符实际上是由第一个操作数所调用的一个方法。你可以重写大部分运算符。

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

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

类型测试操作符

asisis! 操作符在运行时用于检查类型非常方便。

操作符 含义
as 类型转换
is 当对象是相应类型时返回 true
is! 当对象不是相应类型时返回 true

如果obj实现了T所定义的借口,那么obj is T将返回 true。比如,obj is Object必然返回 true。

使用as操作符可以把一个对象转换为特定类型。一般来说,如果在is测试之后还有一些关于对象的表达式,你可以把as当做是is测试的一种简写。考虑下面这段代码:

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

你也可以通过as来简化代码:

(emp as Person).firstName = '永动机';

注意:上面两段代码并不相等。如果emp的值为 null 或者不是一个 Person 对象,第一段代码不会做任何事情,第二段代码会报错 。


赋值操作符

正如你已经看到的,你可以使用=运算符赋值。要仅在变量为null时赋值,请使用??=运算符。

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

诸如+=之类的复合赋值运算符将操作与赋值相结合

= -= /= %= >>= ^=
+= *= ~/= <<= &= |=

以下是复合赋值运算符的工作方式:

  复合赋值 等式表达式
对于操作符op a op b a = a op b
具体例子1 a += b a = a + b
具体例子2 a -= b a = a - b

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

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

逻辑运算符

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

操作符 含义
!expr 反转以下表达式(将false更改为true,反之亦然)
|| 逻辑或
&& 逻辑与

下面是使用逻辑运算符的示例:

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

位运算

通常我们指☞位运算为<<>>移动位运算,通过操作位的移动来达到运算的目的,而&,|,^,~expr也是操作位来达到运算的目的。所以本文统称这些运算都为位运算

操作符 含义
&
|
^ 异或
~expr 一元位补码( 0s变为1s;1s变为0s )
<< 左移
>> 右移

下面是使用所有位运算符的示例:

final value = 0x22;
final bitmask = 0x0f; assert((value & bitmask) == 0x02); // 与
assert((value & ~bitmask) == 0x20); // 与非
assert((value | bitmask) == 0x2f); // 或
assert((value ^ bitmask) == 0x2d); // 异或
assert((value << 4) == 0x220); // 左移
assert((value >> 4) == 0x02); // 右移

条件表达式

dart有两个运算符,可让您简明地评估可能需要if-else语句的表达式:

condition ? expr1 : expr2
如果条件为真,返回expr1,否则返回expr2

expr1 ?? expr2
如果expr1为非空,则返回其值;否则,计算并返回expr2的值。

当你需要根据布尔表达式赋值时,考虑使用?:

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

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

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

前面的例子至少可以用另外两种方式编写,但不像以前那么简洁:

// 稍微长一点的版本使用 ?: 操作符
String playerName(String name) => name != null ? name : 'Guest'; // 非常长的使用if - else语句的版本
String playerName(String name) {
if (name != null) {
return name;
} else {
return 'Guest';
}
}

级联符号(..)

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

示例代码:

querySelector('#confirm') // 获取一个对象
..text = 'Confirm' // 使用它的成员
..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”。

sb.write()调用返回void,你不能在void上构建级联。

注意:严格来说,级联的“双点”符号不是运算符。这只是Dart语法的一部分。


其他操作符

在其他示例中,你已经看到了大多数剩余的运算符:

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

Dart学习-操作符的更多相关文章

  1. [dart学习]第五篇:操作符

    前言:本系列内容假设读者有一定的编程基础,如了解C语言.python等. 本节一起来学习dart的操作符,直接拷贝官网的操作符描述表如下: Description Operator unary pos ...

  2. Dart 学习资料

    Dart 学习资料: 学习资料 网址 Dart 编程语言中文网 http://dart.goodev.org/ Dart 官方包仓库 https://pub.dartlang.org/ 你想了解的Da ...

  3. Dart学习笔记-运算符-条件表达式-类型转换

    Dart学习笔记-运算符-条件表达式-类型转换 一.运算符 1.算术运算符 + (加)- (减)* (乘)/ (除)~/ (取整) %(取余) 2.关系运算符 == (等等) != (不等) > ...

  4. [dart学习]第四篇:函数和操作符(本篇未完待续)

    接着学习dart的函数和操作符. 1.函数Function dart是一种真正的面向对象的语言,通常一个函数也是Function类型的对象,这也就是说可以把函数赋值给一个变量,或者作为另一个函数的入参 ...

  5. 【Dart学习】-- Dart之操作符

    一,概述 dart定义了下表所示的运算符.你可以重写许多这些运算符. 描述 运算符 一元后缀 expr++ expr-- () [] . ?. 一元前缀 -expr !expr ~expr ++exp ...

  6. [dart学习]第三篇:dart变量介绍 (二)

    本篇继续介绍dart变量类型,可参考前文:第二篇:dart变量介绍 (一) (一)final和const类型 如果你不打算修改一个变量的值,那么就把它定义为final或const类型.其中:final ...

  7. [dart学习]第二篇:dart变量介绍 (一)

    前言 本文的所有内容均是官方文档的简单翻译和理解,需要查看原文,请登录  https://www.dartlang.org/guides/language/language-tour  阅读, 让我们 ...

  8. [dart学习]第一篇:windows下安装配置dart编译环境,写出helloworld

    前言 博主非科班出身,平时多用C语言,最近想了解学习一门第二语言,看上了可用于移动开发的目前还小众一点dart,准备用一段比较长的时间来慢慢学习.理解. 关于dart语言不再详细介绍了,大家可以访问  ...

  9. [dart学习]第七篇:类(构造函数)

    前言:楼主平时基本没有使用过异常处理,所以对异常的认知可能不够准确,这里就不翻译异常的相关内容了,大家可以去官网自行阅读介绍,地址 https://dart.dev/guides/language/l ...

随机推荐

  1. 使用TCP通信文件上传

    客服端读取本地文件,吧文件上传到服务器,服务器在吧上传的文件保存到服务器硬盘上方法分析1:客户端使用本地字节输入流读取要上传的文件 2:客户端使用网络字节输出流,吧读取到的文件上传到服务器 3:服务器 ...

  2. Windows 10安装Docker 步骤及顺序

    最近在工作中,重新安装Docker时,遇到了一点坑,故将自己解决经验分享一下~ Hardware assisted virtualization and data execution protecti ...

  3. 修改hostname

    修改hostname步骤 1. 修改/etc/sysconfig/network中的hostname选项 2. 在/etc/hosts中添加hostname对应的ip地址 3.执行命令:hostnam ...

  4. CentOS 7 Squid代理服务器正向代理-传统代理

    Squid是Linux系统中最常用的一款开源代理服务软件,主要提供缓存加速和应用层过滤控制的功能,可以很好的实现HTTP.FTP.DNS查询以及SSL等应用的缓存代理 传统代理:普通的代理服务,多见于 ...

  5. 31 Python中 sys.argv[]的用法简明解释(转)

    Python中 sys.argv[]的用法简明解释 因为是看书自学的python,开始后不久就遇到了这个引入的模块函数,且一直在IDLE上编辑了后运行,试图从结果发现它的用途,然而结果一直都是没结果, ...

  6. SQL中内连接和外连接的区别

    数据表的连接有: 1.内连接(自然连接): 只有两个表相匹配的行才能在结果集中出现 2.外连接: 包括 (1)左外连接(左边的表不加限制) (2)右外连接(右边的表不加限制) (3)全外连接(左右两表 ...

  7. Ubuntu16.04安装Redis

    前言 Redis是常用基于内存的Key-Value数据库,比Memcache更先进,支持多种数据结构,高效,快速.用Redis可以很轻松解决高并发的数据访问问题:作为实时监控信号处理也非常不错. 环境 ...

  8. matlab的Deep Learning的toolbox 中的SAE算法

    最近一直在看Deep Learning,各类博客.论文看得不少 但是说实话,这样做有些疏于实现,一来呢自己的电脑也不是很好,二来呢我目前也没能力自己去写一个toolbox 只是跟着Andrew Ng的 ...

  9. 堆叠式降噪自动编码器(SDA)

    1.1 自动编码器  自动编码器(AutoEncoder,AE)就是一种尽可能复现输入信号的神经网络,其输出向量与输入向量同维,常按照输入向量的某种形式,通过隐层学习一个数据的表示或对原始数据进行有效 ...

  10. 使用4K分辨率,然后放大DIP200%,软件界面异常.

    简单:WFM主界面.AutoScaleMode  选中DIP,然后使用表格容器,容器分割,容器.就可以快速迁移旧程序. 复杂点:读取桌面分辨率,DIP放大....