dart系列之:dart中的異步編程
簡介
熟悉javascript的朋友應(yīng)該知道,在ES6中引入了await和async的語法,可以方便的進(jìn)行異步編程,從而擺脫了回調(diào)地獄。dart作為一種新生的語言,沒有理由不繼承這種優(yōu)秀的品質(zhì)。很自然的,dart中也有await和async語言,一起來看看吧。
為什么要用異步編程
那么為什么要用異步編程呢? 只用同步不能夠解決嗎?
其實大多情況下同步已經(jīng)夠用了,但是在下面的幾種情況下,同步的場景還是有缺陷的。
- 需要花很長時間從網(wǎng)絡(luò)上下載數(shù)據(jù)的情況。
- 讀取數(shù)據(jù)庫的耗時情況。
- 從文件讀取數(shù)據(jù)的情況。
總結(jié)而言,如果某些操作需要花費大量的時間,那么就可以用到異步編程了。
怎么使用
async是方法的描述符,如果要使用await,則必須配合async一起使用:
Future<void> checkVersion() async { var version = await lookUpVersion(); // Do something with version }
注意,await后面一般接著的是Future對象。
先看一個錯誤使用異步編程的例子:
String createOrderMessage() { var order = fetchUserOrder(); return 'Your order is: $order'; } Future<String> fetchUserOrder() => Future.delayed( const Duration(seconds: 2), () => 'Order one!', ); void main() { print(createOrderMessage()); }
上面的代碼本意是打印出從數(shù)據(jù)庫耗時取出的數(shù)據(jù),但是結(jié)果并不是想象的那樣,其原因就是fetchUserOrder方法是一個異步方法,所以不會立即返回,從而導(dǎo)致結(jié)果打印失敗。
將上面的代碼用async改寫:
Future<String> createOrderMessage() async { var order = await fetchUserOrder(); return 'Your order is: $order'; } Future<String> fetchUserOrder() => Future.delayed( const Duration(seconds: 2), () => 'Large Latte', ); Future<void> main() async { print('Fetching user order...'); print(await createOrderMessage()); }
Future
上面我們在使用async和await的過程中使用到了Future。在java中Future表示的是線程的執(zhí)行結(jié)果。在dart中Future表示的是一個異步執(zhí)行的結(jié)果。
Future有兩種狀態(tài):uncompleted 或者 completed。
當(dāng)最開始執(zhí)行一個異步函數(shù)的時候,會返回一個未完成的Future。這個未完成的Future會等等異步執(zhí)行的完成或者失敗。
不管異步程序是成功還是失敗,最終都會返回一個完成狀態(tài)。
async返回的Future可以接泛型,表示的時候返回的具體類型,比如Future 表示的是返回一個字符串,而 Future表示不返回任何值。
下面是兩個不同返回的例子:
Future<String> fetchUserOrder() { return Future.delayed(const Duration(seconds: 2), () => 'Large Latte'); } Future<void> fetchUserOrder() { return Future.delayed(const Duration(seconds: 2), () => print('Large Latte')); }
下面是一個異常的例子:
Future<void> fetchUserOrder() { return Future.delayed(const Duration(seconds: 2), () => throw Exception('Logout failed: user ID is invalid')); }
異步異常處理
在async的函數(shù)中,對await的異步方法中拋出的異常,可以直接是用try catch來進(jìn)行異常的捕獲:
try { print('Awaiting user order...'); var order = await fetchUserOrder(); } catch (err) { print('Caught error: $err'); }
在同步函數(shù)中調(diào)用異步函數(shù)
上面介紹的fetchUserOrder()返回的是一個Future,表示的是一個異步執(zhí)行的過程。
那么如果是一個同步的方法,比如main()函數(shù)中,如何去調(diào)用異步方法,并且得到返回值呢?
await肯定是不行的,因為await只能在async的方法中調(diào)用。這個時候就可以用到then語句:
fetchUserOrder().then(order=>'do something');
then語句會等待異步執(zhí)行返回結(jié)果,然后對結(jié)果進(jìn)行處理,實際上就等同于javascript中的回調(diào)。
總結(jié)
以上就是dart中async和await的用法。