Flutter - Future与FutureBuilder实用技巧

我们知道Dart语言的执行环境是”单线程”。也就是指一次只能完成一件任务。如果有多个任务,就必须排队,前面一个任务完成,再执行后面一个任务。

为了解决这个问题,Dart语言将任务的执行模式分成两种:同步和异步。

异步:Future与FutureBuilder实用技巧

  • 什么是Future?
  • Future的常见用法?
    • 获取Future的结果?
    • 捕获Future的异常?
    • 结合async,await?
    • future.whenComplete?
    • future.timeout?
  • 什么是FutureBuilder?
  • FutureBuilder常见的用法?

什么是Future?

Future表示在接下来的某个时间的值或错误,借助Future我们可以在Flutter实现异步操作。

它类似于ES6中的Promise,提供thencatchError的链式调用;

Futuredart:async包中的一个类,使用它时需要导入dart:async包,Future有两种状态:

  • pending - 执行中;
  • completed - 执行结束,分两种情况要么成功要么失败;

Future的常见用法?

  • 使用future.then获取future的值与捕获future的异常
  • 结合async,await
  • future.whenComplete
  • future.timeout

使用future.then获取future的值与捕获future的异常

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
import 'dart:async';

Future<String> testFuture() {
// throw new Error();
return Future.value('success');
// return Future.error('error');
}

main() {
testFuture().then((s) {
print(s);
}, onError: (e) {
print('onError:');
print(e);
}).catchError((e) {
print('catchError:');
print(e);
});
}

如果catchError与onError同时存在,则会只调用onError;

Future的then`的原型:

1
Future<R> then<R>(FutureOr<R> onValue(T value), {Function onError});

第一个参数会成功的结果回调,第二个参数onError可选表示执行出现异常。

结合async await

Future是异步的,如果我们要将异步转同步,那么可以借助async await来完成。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
import 'dart:async';

test() async {
int result = await Future.delayed(Duration(milliseconds: 2000), () {
return Future.value(123);
});
print('t3:' + DateTime.now().toString());
print(result);
}

main() {
print('t1:' + DateTime.now().toString());
test();
print('t2:' + DateTime.now().toString());
}

future.whenComplete

有时候我们需要在Future结束的时候做些事情,我们知道then().catchError()的模式类似于try-catchtry-catch有个finally代码块,而future.whenComplete就是Future的finally。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
import 'dart:async';
import 'dart:math';

void main() {
var random = Random();
Future.delayed(Duration(seconds: 3), () {
if (random.nextBool()) {
return 100;
} else {
throw 'boom!';
}
}).then(print).catchError(print).whenComplete(() {
print('done!');
});
}

future.timeout

完成一个异步操作可能需要很长的时间,比如:网络请求,但有时我们需要为异步操作设置一个超时时间,那么,如何为Future设置超时时间呢?

1
2
3
4
5
6
7
import 'dart:async';

void main() {
new Future.delayed(new Duration(seconds: 3), () {
return 1;
}).timeout(new Duration(seconds: 2)).then(print).catchError(print);
}

运行上述代码会看到:TimeoutException after 0:00:02.000000: Future not completed

什么是FutureBuilder?

FutureBuilder是一个将异步操作和异步UI更新结合在一起的类,通过它我们可以将网络请求,数据库读取等的结果更新的页面上。

FutureBuilder的构造方法

1
FutureBuilder({Key key, Future<T> future, T initialData, @required AsyncWidgetBuilder<T> builder })
  • future: Future对象表示此构建器当前连接的异步计算;
  • initialData: 表示一个非空的Future完成前的初始化数据;
  • builder: AsyncWidgetBuilder类型的回到函数,是一个基于异步交互构建widget的函数;

这个builder函数接受两个参数BuildContext context 与 AsyncSnapshot snapshot,它返回一个widget。AsyncSnapshot包含异步计算的信息,它具有以下属性:

  • connectionState - 枚举ConnectionState的值,表示与异步计算的连接状态,ConnectionState有四个值:none,waiting,active和done;
  • data - 异步计算接收的最新数据;
  • error - 异步计算接收的最新错误对象;

AsyncSnapshot还具有hasData和hasError属性,以分别检查它是否包含非空数据值或错误值。

现在我们可以看到使用FutureBuilder的基本模式。 在创建新的FutureBuilder对象时,我们将Future对象作为要处理的异步计算传递。 在构建器函数中,我们检查connectionState的值,并使用AsyncSnapshot中的数据或错误返回不同的窗口小部件。

https://flutter-academy.com/async-in-flutter-futurebuilder/
FutureBuilder的使用?

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
import 'dart:convert';

import 'package:flutter/material.dart';
import 'package:http/http.dart' as http;

void main() => runApp(new MyApp());

class MyApp extends StatefulWidget {
@override
State<StatefulWidget> createState() => _MyAppState();
}

class _MyAppState extends State<MyApp> {
String showResult = '';

Future<CommonModel> fetchPost() async {
final response = await http
.get('http://www.devio.org/io/flutter_app/json/test_common_model.json');
Utf8Decoder utf8decoder = Utf8Decoder(); //fix 中文乱码
var result = json.decode(utf8decoder.convert(response.bodyBytes));
return CommonModel.fromJson(result);
}

@override
Widget build(BuildContext context) {
return MaterialApp(
home: Scaffold(
appBar: AppBar(
title: Text('Future与FutureBuilder实用技巧'),
),
body: FutureBuilder<CommonModel>(
future: fetchPost(),
builder:
(BuildContext context, AsyncSnapshot<CommonModel> snapshot) {
switch (snapshot.connectionState) {
case ConnectionState.none:
return new Text('Input a URL to start');
case ConnectionState.waiting:
return new Center(child: new CircularProgressIndicator());
case ConnectionState.active:
return new Text('');
case ConnectionState.done:
if (snapshot.hasError) {
return new Text(
'${snapshot.error}',
style: TextStyle(color: Colors.red),
);
} else {
return new Column(children: <Widget>[
Text('icon:${snapshot.data.icon}'),
Text('statusBarColor:${snapshot.data.statusBarColor}'),
Text('title:${snapshot.data.title}'),
Text('url:${snapshot.data.url}')
]);
}
}
}),
),
);
}
}

class CommonModel {
final String icon;
final String title;
final String url;
final String statusBarColor;
final bool hideAppBar;

CommonModel(
{this.icon, this.title, this.url, this.statusBarColor, this.hideAppBar});

factory CommonModel.fromJson(Map<String, dynamic> json) {
return CommonModel(
icon: json['icon'],
title: json['title'],
url: json['url'],
statusBarColor: json['statusBarColor'],
hideAppBar: json['hideAppBar'],
);
}
}