异步支持
🎉🎉🎉Dart 代码库中有大量返回 Future
或 Stream
对象的函数,这些函数都是 异步
的,它们会在耗时 *** 作**(比如I/O)**执行完毕前直接返回而不会等待耗时 *** 作执行完毕。
🎉🎉🎉async
和 await
关键字用于实现异步编程,并且让你的代码看起来就像是同步的一样。
async
和 await
的代码是异步的,但是看起来有点像同步代码。例如,下面的代码使用 await 等待异步函数的执行结果。
await lookUpVersion();
必须在带有 async 关键字的 异步函数 中使用 await:
Future<void> checkVersion() async {
var version = await lookUpVersion();
// Do something with version
}
使用 try、catch 以及 finally 来处理使用 await 导致的异常:尽管异步函数可以处理耗时 *** 作,但是它并不会等待这些耗时 *** 作完成,异步函数执行时会在其遇到第一个 await 表达式(代码行)时返回一个 Future 对象,然后等待 await 表达式执行完毕后继续执行。
try {
version = await lookUpVersion();
} catch (e) {
// React to inability to look up the version
}
你可以在异步函数中多次使用 await 关键字。例如,下面代码中等待了三次函数结果:
var entrypoint = await findEntryPoint();
var exitCode = await runExecutable(entrypoint, args);
await flushThenExit(exitCode);
await 表达式的返回值通常是一个 Future 对象;如果不是的话也会自动将其包裹在一个 Future 对象里。 Future 对象代表一个“承诺”, await 表达式会阻塞直到需要的对象返回。
如果在使用 await 时导致编译错误,请确保 await 在一个异步函数中使用。
例如,如果想在 main() 函数中使用 await,那么 main() 函数就必须使用 async 关键字标识。void main() async {
checkVersion();
print('In main: version is ${await lookUpVersion()}');
}
2、声明异步函数如上的例子使用了声明为 async 的函数 checkVersion(),但没有等待其结果。在实际的开发中,如果代码假设函数已经执行完成,则可能导致一些异步的问题。
异步函数 是函数体由 async 关键字标记的函数。
将关键字async
添加到函数并让其返回一个 Future
对象。假设有如下返回 String
对象的方法:
String lookUpVersion() => '1.0.0';
将其改为异步函数,返回值是 Future:
Future<String> lookUpVersion() async => '1.0.0';
注意,函数体不需要使用 Future API。如有必要,Dart 会创建 Future 对象。
如果函数没有返回有效值,需要设置其返回类型为 Future。
3、async/await 使用async
和await
的代码是异步的,但是看起来很像同步代码。当我们需要获得A的结果,再执行B,时,你需要then()->then()
,但是利用async
与await
能够非常好的解决回调地狱的问题:
//async 表示这是一个异步方法,await必须再async方法中使用
//异步方法只能返回 void和Future
Future<String> readFile() async {
//await 等待future执行完成再执行后续代码
String content = await new File("/Users/xiang/enjoy/a.txt").readAsString();
String content2 = await new File("/Users/xiang/enjoy/a.txt").readAsString();
//自动转换为 future
return content;
}
4、处理 Stream
如果想从 Stream 中获取值,可以有两种选择:
使用async
关键字和一个 异步循环(使用 await for
关键字标识)。使用 Stream API。
使用 await for 定义异步循环看起来是这样的:在使用
await for
关键字前,确保其可以令代码逻辑更加清晰并且是真的需要等待所有的结果执行完毕。例如,通常不应该在 UI 事件监听器上使用await for
关键字,因为 UI 框架发出的事件流是无穷尽的。
await for (varOrType identifier in expression) {
// Executes each time the stream emits a value.
}
表达式 的类型必须是 Stream。执行流程如下:
等待直到 Stream 返回一个数据。使用 1 中 Stream 返回的数据执行循环体。重复 1、2 过程直到 Stream 数据返回完毕。使用 break
和 return
语句可以停止接收 Stream 数据,这样就跳出了循环并取消注册监听 Stream。
如果在实现异步 for 循环时遇到编译时错误,请检查确保 await for 处于异步函数中。 例如,要在应用程序的 main() 函数中使用异步 for 循环,main() 函数体必须标记为 async:
void main() async {
// ...
await for (final request in requestServer) {
handleRequest(request);
}
// ...
}
5、isolate机制
Dart是基于单线程模型的语言。但是在开发当中我们经常会进行耗时 *** 作比如网络请求,这种耗时 *** 作会堵塞我们的代码,所以在Dart也有并发机制,名叫isolate。APP的启动入口main
函数就是一个类似Android主线程的一个主isolate。和Java的Thread不同的是,Dart中的isolate无法共享内存。
import 'dart:isolate';
int i;
void main() {
i = 10;
//创建一个消息接收器
ReceivePort receivePort = new ReceivePort();
//创建isolate
Isolate.spawn(isolateMain, receivePort.sendPort);
//接收其他isolate发过来的消息
receivePort.listen((message) {
//发过来sendPort,则主isolate也可以向创建的isolate发送消息
if (message is SendPort) {
message.send("好呀好呀!");
} else {
print("接到子isolate消息:" + message);
}
});
}
/// 新isolate的入口函数
void isolateMain(SendPort sendPort) {
// isolate是内存隔离的,i的值是在主isolate定义的所以这里获得null
print(i);
ReceivePort receivePort = new ReceivePort();
sendPort.send(receivePort.sendPort);
// 向主isolate发送消息
sendPort.send("去大保健吗?");
receivePort.listen((message) {
print("接到主isolate消息:" + message);
});
}
6、 生成器
当你需要延迟地生成一连串的值时,可以考虑使用 生成器函数。Dart 内置支持两种形式的生成器方法:
同步 生成器:返回一个Iterable
对象。异步 生成器:返回一个 Stream
对象。
通过在函数上加 sync* 关键字并将返回值类型设置为
Iterable
来实现一个 同步 生成器函数,在函数中使用yield
语句来传递值:
Iterable<int> naturalsTo(int n) sync* {
int k = 0;
while (k < n) yield k++;
}
实现 异步 生成器函数与同步类似,只不过关键字为
async
* 并且返回值为 Stream:
Stream<int> asynchronousNaturalsTo(int n) async* {
int k = 0;
while (k < n) yield k++;
}
如果生成器是递归调用的,可是使用 yield
* 语句提升执行性能:
Iterable<int> naturalsDownFrom(int n) sync* {
if (n > 0) {
yield n;
yield* naturalsDownFrom(n - 1);
}
}
7、元数据
使用元数据可以为代码增加一些额外的信息。元数据注解以 @ 开头,其后紧跟一个编译时常量(比如 deprecated)或者调用一个常量构造函数。
Dart 中有两个注解是所有代码都可以使用的:@deprecated、@Deprecated
和 @override
。你可以查阅 扩展一个类 获取有关 @override
的使用示例。下面是使用 @deprecated 的示例:
class Television {
/// Use [turnOn] to turn the power on instead.
@Deprecated('Use turnOn instead')
void activate() {
turnOn();
}
/// Turns the TV's power on.
void turnOn() {...}
// ···
}
可以自定义元数据注解。下面的示例定义了一个带有两个参数的 @todo
注解:
library todo;
class Todo {
final String who;
final String what;
const Todo(this.who, this.what);
}
使用 @Todo
注解的示例:
import 'todo.dart';
@Todo('seth', 'make this do something')
void doSomething() {
print('do something');
}
元数据可以在 library、class、typedef、type parameter、 constructor、factory、function、field、parameter 或者 variable 声明之前使用,也可以在 import 或 export 之前使用。可使用反射在运行时获取元数据信息。
总结🎉🎉🎉 都看到这里了,能不能留个赞或者是评论呢?😚😚😚
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)