Dart编程语言学习之操作符、控制流和异常处理

2023年 7月 13日 42.2k 0

操作符

Dart 操作符和主流语言的操作符类似, 只要有一门常用语言, 对 Dart 掌握也是很快的

Dart 和以前介绍的 Kotlin 类似, 也提供操作符重载功能

算术操作符

Dart 支持下面常用的数学操作符:

操作符 意义
+ 加号
- 减号
-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
复制代码

需要注意的是两个整型相除是返回一个 double 类型的值,而不是像 Java 一样返回一个整型,如果需要返回整型可以使用 ~/

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; // 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
复制代码

比较操作符

操作符 意义
== 相等
!= 不等
> 大于
= 大于等于
>= a >>= b 和 a = a >> b 等价
^= a ^= b 和 a = a ^ b 等价
+= a += b 和 a = a + b 等价
*= a *= b 和 a = a * b 等价
~/= a ~/= b 和 a = a ~/ b 等价
Vector(x + v.x, y + v.y);

// 重载 - 操作符
Vector operator -(Vector v) => Vector(x - v.x, y - v.y);

// 重载 == 操作符
bool operator ==(other) => other is Vector
&& runtimeType == other.runtimeType
&& x == other.x && y == other.y;

}

void main() {
final v = Vector(2, 3);
final w = Vector(2, 2);

assert(v + w == Vector(4, 5));
assert(v - w == Vector(0, 1));
}

复制代码

如果我们重载了 == 操作符, 也要重载 hashCode 函数, 类似 Java 中重载了 equals 函数,最好也重载 hashCode 函数是一样的, 因为对象的 hash 值决定对象的存储位置

@override
int get hashCode => x.hashCode ^ y.hashCode;
复制代码

控制流语句

if/else

Dart 支持 if 语句和可选的 else 语句

if (isRaining()) {
    print("raining");
} else if (isSnowing()) {
    print("snowing");
} else {
    print("unknown");
}
复制代码

for循环

Dart 不仅支持标准的 for 循环:

for (var i = 0; i < 5; i++) {
}
复制代码

还支持 for-in 那些实现了 Iterator 接口的类(如List/Set):

var collection = [0, 1, 2];
for (var x in collection) {
  print(x); // 0 1 2
}
复制代码

通过上一篇文章(三)Flutter学习之Dart函数的介绍知道

Dart Closures 能够访问自身作用域内的变量, 哪怕这个变量是外部传递给 Closure 的 如:

var callbacks = [];
for (var i = 0; i  print(i));
}
callbacks.forEach((c) => c());

// 输出
1
2
复制代码

但是在 JavaScript 中会输出两个 2

while/do-while/break/continue

while/do-while/break/continue 和其他语言没有什么区别, 在这里就不赘述了

switch-case

Switch 语句可以接受 整型、字符串 、枚举 或者 编译时常量, 然后使用 == 进行比较, 下面看下常量的比较:

class Person {
  final String id;

  const Person(this.id);
}

const p = Person("001");
const p1 = Person("001");
const p2 = Person("003");
  
switch (p) {
    case p1:
      print(p1.id);
      break;
    case p2:
      print(p2.id);
      break;
    default:
      print("unknown");
  }
复制代码

如果 case 子句不是空, 要以 break/return/throw/continue 结尾, 否则会报错

如果想要 case 子句之间 fall-through 的话, 将 case 子句为空即可

  var command = 'CLOSED';
  switch (command) {
    case 'CLOSED':
    case 'NOW_CLOSED':
      print("NOW_CLOSED");
      break;

    default:
      print("UNKNOWN");
  }
复制代码

还可以使用 continue 的方式 fall-through

var command = 'CLOSED';
switch (command) {
case 'CLOSED':
  print("CLOSED");
  continue nowClosed;

nowClosed:
case 'NOW_CLOSED':
  print("NOW_CLOSED");
  break;

default:
  print("UNKNOWN");
}

// 输出

CLOSED
NOW_CLOSED

复制代码

断言(assert)

在开发阶段, 我们可以使用断言语句 assert(condition, optionalMessage); 来中断程序的正常执行, 当 conditionfalse 的时候(抛出 AssertionError 异常); 如果 conditiontrue, 则继续执行程序的下一行代码. 例如:

// Make sure the value is less than 100.
assert(number < 100);

// Make sure this is an https URL.
assert(urlString.startsWith('https'));

assert(urlString.startsWith('https'),
    'URL ($urlString) should start with "https".');
复制代码

断言什么时候生效呢?这取决于你使用的工具和框架:

  • Flutter 在 debug 模式启用断言
  • 仅开发阶段使用的开发工具如 dartdevc, 默认是开启断言
  • 诸如 dartdart2js 等工具支持命令行来启用断言: --enable-asserts

在生产环境的代码, 断言语句将会被忽略, 断言的 condition 表达式不会被执行,所以不用担心性能问题

异常处理

Dart 提供了 throw, rethrow, try, catch, on, finally 关键字让开发者能够抛出和捕获异常

Java 不同的是, Dart 中所有的异常都是 unchecked 异常, 也就是说编译器不会强制开发者去捕获任何异常, 除非你有这个需要

Dart 提供了两个类异常: ExceptionError, Dart 不仅可以抛出异常还可以抛出任何不为 null 的对象:

// 抛出异常
throw FormatException('Expected at least 1 section');

// 抛出不为 null 的对象
throw 'Out of llamas!';
复制代码

虽然 Dart 允许我们抛出一个不为 null 的普通对象,但是官方还是建议我们抛出的异常继承自 ExceptionError

介绍完了 throw 关键字,我们来看下 catchon 和 关键字:

catchon 都是用来捕获异常:

try {
  breedMoreLlamas();
} on OutOfLlamasException {
  // 捕获一个特定的异常
  buyMoreLlamas();
} on Exception catch (e) {
  // 捕获所有继承自 Exception 的异常,并拿到异常对象
  print('Unknown exception: $e');
} catch (e) {
  // 捕获所有异常
  print('Something really unknown: $e');
}

复制代码

可见, on 关键字用于指定捕获特定的异常, catch 关键字用于拿到异常对象

catch 关键字除了可以拿到异常对象, 还可以拿到异常的 堆栈 信息, 如:

try {
  // ···
} on Exception catch (e) {
  print('Exception details:\n $e');
} catch (e, s) {
  print('Exception details:\n $e');
  print('Stack trace:\n $s');
}

复制代码

一般情况下, 使用了 on, catch 关键字来捕获异常, 异常会停止传播, 如果需要异常继续传播可以使用 rethrow 关键字

void misbehave() {
  try {
    dynamic foo = true;
    print(foo++); // Runtime error
  } catch (e) {
    print('misbehave() partially handled ${e.runtimeType}.');
    rethrow; // Allow callers to see the exception.
  }
}

void main() {
  try {
    misbehave();
  } catch (e) {
    print('main() finished handling ${e.runtimeType}.');
  }
}

复制代码

最后介绍 Dart 异常处理的最后一个关键字 finally

finnaly 关键字很简单 , 就是不管是否抛出异常 finally 子句一定会执行:

try {
  breedMoreLlamas();
} finally {
  // 就算抛出异常(程序中断执行), finnaly 也会先执行
  cleanLlamaStalls();
}

try {
  breedMoreLlamas();
} catch (e) {
  // 捕获异常
  print('Error: $e');
} finally {
  // 执行 finally 子句
  cleanLlamaStalls(); // Then clean up.
}
复制代码

关于 Dart操作符, 控制流, 异常处理 就介绍完毕

相关文章

JavaScript2024新功能:Object.groupBy、正则表达式v标志
PHP trim 函数对多字节字符的使用和限制
新函数 json_validate() 、randomizer 类扩展…20 个PHP 8.3 新特性全面解析
使用HTMX为WordPress增效:如何在不使用复杂框架的情况下增强平台功能
为React 19做准备:WordPress 6.6用户指南
如何删除WordPress中的所有评论

发布评论