Flutter/Dart第12天:Dart控制流详解

Dart控制流主要由循环分支组成:

重要说明:本博客基于Dart官网文档,但并不是简单的对官网进行翻译,在覆盖核心功能情况下,我会根据个人研发经验,加入自己的一些扩展问题和场景验证。

循环

Dart可通过循环来控制逻辑流,它支持3种语句:for循环,whiledo while循环,breakcontinue

同时,也可以通过非循环来控制逻辑流,包括:分支(如:ifswitch)和异常(trycatchthrow)等

for循环

for循环可以使用在迭代中,标准用法如下代码样例:

var message = StringBuffer('Dart is fun');
for (var i = 0; i < 5; i++) {
 message.write('!');
}

for-in循环常用于无需关注迭代的索引,如下代码样例,包含了闭包列表和迭代:

var callbacks = [];
for (var i = 0; i < 2; i++) {
 callbacks.add(() => print(i));
}
for (final c in callbacks) {
 c();
}
// 解构:01

同样的,for-in循环还可以使用对象模式(Pattern模式详解:https://ntopic.cn/p/2023100501),解构迭代的元素:

for (final Candidate(:name, :yearsExperience) in candidates) {
 print('$name has $yearsExperience of experience.');
}

对于迭代类型(Dart可迭代集合详解:https://ntopic.cn/p/2023092701),forEach()方法也可以实现循环:

var collection = [1, 2, 3];
collection.forEach(print); // 1 2 3

while和do while循环

while循环do while循环最大的区别:while先检测循环条件,然后执行循环逻辑;do while默认先执行1次循环逻辑,然后检测循环条件,以决定后续是否继续执行循环逻辑。

while (!isDone()) {
 doSomething();
}
do {
 printLine();
} while (!atEndOfPage());

break和continue循环控制

breakcontinue在循环中的最大区别:break中断整个循环,而continue中断当前循环的后续逻辑

while (true) {
 if (shutDownRequested()) break;
 processIncomingRequests();
}
for (int i = 0; i < candidates.length; i++) {
 var candidate = candidates[i];
 if (candidate.yearsExperience < 5) {
 continue;
 }
 candidate.interview();
}
// 若`candidates`是可迭代列表,则上面的逻辑可改写如下:
candidates
 .where((c) => c.yearsExperience >= 5)
 .forEach((c) => c.interview());

分支

Dart也可通过分支来控制逻辑流:if语句和元素,if-case语句和元素,switch语句和switch表达式等。

同样的,我们也可以通过其他方式来控制逻辑流:如循环(就是本文的第1节)和异常(trycatchthrow)等

if/else分支

if语句可以有else可选项子句,if后面的括号中的表达式计算值必须是boolean类型,如下代码样例:

if (isRaining()) {
 you.bringRainCoat();
} else if (isSnowing()) {
 you.wearJacket();
} else {
 car.putTopDown();
}

if-case模式

Dart中if语句后的case子句可以带一个模式(Pattern模式详解:https://ntopic.cn/p/2023100501):

if (pair case [int x, int y]) return Point(x, y);

上诉代码样例中,列表模式[int x, int y]首先匹配pair值,并且把列表第1和第2个int类型的元素解构并赋值给了xy变量。

如果模式未能成功匹配,则需要进入else分支:

if (pair case [int x, int y]) {
 print('Was coordinate array $x,$y');
} else {
 throw FormatException('Invalid coordinates.');
}

switch语句和switch表达式

switch语句后面可以有多个case子句,每一个case子句都可以是一个模式(Pattern模式详解:https://ntopic.cn/p/2023100501)。

当没有case子句匹配时,就会执行default子句或者_通配符子句:

switch (command) {
 case 'OPEN':
 executeOpen();
 continue newCase; // 继续执行
 case 'DENIED':
 case 'CLOSED':
 executeClosed(); // `DENIED`和`CLOSED`均会执行
 newCase:
 case 'PENDING':
 executeNowClosed(); // `OPEN`和`PENDING`均会执行
}

在switch语句中,还可以使用模式,有如下几种用法:

var x = switch (y) { ... };
print(switch (x) { ... });
return switch (x) { ... };

switch表达式可以重写switch语句,如下为switch语句的代码样例:

// `slash`, `star`, `comma`, `semicolon`等,需要是常量
switch (charCode) {
 case slash || star || plus || minus: // `逻辑或`模式
 token = operator(charCode);
 case comma || semicolon: // `逻辑或`模式
 token = punctuation(charCode);
 case >= digit0 && <= digit9: // `关系`和`逻辑与`模式
 token = number();
 default:
 throw FormatException('Invalid');
}

上诉代码样例,可以使用switch表达式重写如下:

token = switch (charCode) {
 slash || star || plus || minus => operator(charCode),
 comma || semicolon => punctuation(charCode),
 >= digit0 && <= digit9 => number(),
 _ => throw FormatException('Invalid')
};

switch表达式不同于switch语句的语法,包括以下几个方面:

  • Case可选项无需使用case开头。
  • Case可选项的逻辑是一个表达式,而不是一系列的语句。
  • Case可选项都必须有逻辑,空可选性不代表隐性失败。
  • Case可选项模式,逻辑使用=>分割。
  • 多个Case可选项之间,使用,分割。
  • 默认可选项,只能使用_

Switch还可以使用在穷举检测中,就是所有的值都有对应的可选项被处理,以避免引发编译错误。默认可选项(default_)涵盖了所有其他情况,因此它可以保障穷举性。

case-when子句

case-when子句:case子句后面增加when子句。case-when子句可以用在if-caseswitch语句switch表达式中。

switch (pair) {
 case (int a, int b) when a > b:
 print('First element greater');
 case (int a, int b):
 print('First element not greater');
}

case-when子句计算任何一个boolean类型的值,值为true代表可以执行本Case可选项逻辑,为false继续执行下一个Case可选项,并不会退出整个Switch语句。


我的本博客原地址:https://ntopic.cn/p/2023101301


作者:老牛啊原文地址:https://www.cnblogs.com/obullxl/p/NTopic2023101301.html

%s 个评论

要回复文章请先登录注册