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. 【开源】.net 分布式架构之监控平台

    开源地址:http://git.oschina.net/chejiangyi/Dyd.BaseService.Monitor .net 简单监控平台,用于集群的性能监控,应用耗时监控管理,统一日志管理 ...

  2. 一次Oracle 11g+FluentNHibernate AutoMapping组合尝试

    本文的上下文环境 操作系统:Win7 x64 Ultimate开发工具:Visual Studio 2013 一.前言 在以前开发的大多数场景下,使用的开发语言是C#,使用的数据库是Sql Serve ...

  3. About_datebase

    1:创建数据库的名字 create database + 表名; 2:连接数据库 use + 表名; 3:删除数据库 drop database + 表名; 4:创建表 create table + ...

  4. Android如何在java代码中设置margin

    习惯了直接在xml里设置margin(距离上下左右都是10dip),如: <ImageView android:layout_margin="10dip" android:s ...

  5. iOS开发--泛型

    一. 泛型函数 1.单一占位符泛型函数 下面就使用一个经典案例:两个数值进行交换.来使用泛型,写一个通用的函数,这个函数的功能就是交换两个变量的值.在Swift中不允许类型隐式转换,也就是说,如果你定 ...

  6. 在Windows基础上(硬盘)安装Linux操作系统(CentOS/RedHat)

    注:该方法安装CentOS ,RedHat均没有问题,其它Linux操作系统,没有尝试过. 0.创建一个fat32的盘.我分了8G给这个盘,盘符为F.F盘以后的内存所有删除,作为未分配的内存.这个留用 ...

  7. 2017-02-19C#基础 - 数据类型与类型转换

    数据类型 基本数据类型 1)整形:byte  short  int  long 整数类型 2)浮点型:fioat(.NET类型 Single 值后面要加f float = 10.5f;)  doubl ...

  8. linux应用态下的时间

    1.时间值 1.1 日历时间(UTC) 该值是自1 9 7 0年1月1日0 0 : 0 0 : 0 0以来国际标准时间( U T C)所经过的秒数累计值(早期的手册称 U T C为格林尼治标准时间) ...

  9. Linux下导入SQL文件

    导入数据库 一.首先建空数据库 格式: mysql>create database 数据库名;举例: mysql>create database abc; 二.导入数据库 方法一: 选择数 ...

  10. java RMI原理详解

    java本身提供了一种RPC框架——RMI(即Remote Method Invoke 远程方法调用),在编写一个接口需要作为远程调用时,都需要继承了Remote,Remote 接口用于标识其方法可以 ...