1.异步方法的实现原理
异步方法不需要多线程,因为一个异步方法并不是运行在一个独立的线程中的。
异步方法运行在当前同步上下文中,只有激活的时候才占用当前线程的时间。
异步模型采用时间片轮转来实现。
2.使用异步编程模型的优势:
避免性能瓶颈,提升应用程序的整体响应性。
3.关键字
全新的异步编程模型使用“async”和“await”关键字来编写异步方法
async:用来标识一个方法,lambda表达式,或者一个匿名方法是异步的;
await:用来标识一个异步方法应该在此处挂起执行,直到等待的任务完成,于此同时,控制权会移交给异步方法的调用方。
4.异步方法的参数和返回值
异步方法的参数: 不能使用“ref”参数和“out”参数,但是在异步方法内部可以调用含有这些参数的方法
异步方法的返回类型:
Task<TResult>:Tresult为异步方法的返回值类型。
Task:异步方法没有返回值。
void:主要用于事件处理程序(不能被等待,无法捕获异常)。
5.异步方法的命名规范
*异步方法的方法名应该以Async作为后缀
*事件处理程序,基类方法和接口方法,可以忽略此命名规范:
*例如: startButton_Click不应重命名为startButton_ClickAsync
6.一个Demo
1 <Window
2 xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
3 xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
4 xmlns:d="http://schemas.microsoft.com/expression/blend/2008" xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006" mc:Ignorable="d" x:Class="AsyncSample.MainWindow"
5 Title="Control Flow Trace" Height="350" Width="592">
6 <Grid>
7 <Button x:Name="startButton" Content="Start
" HorizontalAlignment="Left" Margin="250,10,0,0" VerticalAlignment="Top" Width="75" Height="24" Click="startButton_Click" d:LayoutOverrides="GridBox"/>
8 <TextBox x:Name="resultsTextBox" HorizontalAlignment="Left" TextWrapping="Wrap" VerticalAlignment="Bottom" Width="576" Height="265" FontFamily="Lucida Console" FontSize="10" VerticalScrollBarVisibility="Visible" Grid.ColumnSpan="3"/>
9 </Grid>
10 </Window>
1 using System;
2 using System.Collections.Generic;
3 using System.Linq;
4 using System.Text;
5 using System.Threading.Tasks;
6 using System.Windows;
7 using System.Windows.Controls;
8 using System.Windows.Data;
9 using System.Windows.Documents;
10 using System.Windows.Input;
11 using System.Windows.Media;
12 using System.Windows.Media.Imaging;
13 using System.Windows.Navigation;
14 using System.Windows.Shapes;
15 using System.Net.Http;
16
17 namespace AsyncSample
18 {
19 /// <summary>
20 /// MainWindow.xaml 的交互逻辑
21 /// </summary>
22 public partial class MainWindow : Window
23 {
24 public MainWindow()
25 {
26 InitializeComponent();
27
28 }
29 private async void startButton_Click(object sender, RoutedEventArgs e)
30 {
31 // 1
32 Task<string> getLengthTask = AccessTheWebAsync();
33
34 // 4
35 string contentLength = await getLengthTask;
36
37 // 6
38 resultsTextBox.Text +=
39 String.Format("\r\nLength of the downloaded string: {0}.\r\n", contentLength);
40 }
41
42
43 async Task<string> AccessTheWebAsync()
44 {
45 // 2
46 HttpClient client = new HttpClient();
47 Task<string> getStringTask =
48 client.GetStringAsync(http://www.cnblogs.com);
49
50 // 3
51 string urlContents = await getStringTask;
52
53 // 5
54 return urlContents;
55 }
56 }
57 }
7.异步方法的执行序列
1: 进入startButton_Click方法
调用 AccessTheWebAsync.
2: 进入 AccessTheWebAsync
调用HttpClient.GetStringAsync.
3: 回到 AccessTheWebAsync
任务getStringTask开始.
等待getStringTask & 返回一个Task<int>实例给startButton_Click.
4: 回到startButton_Click
任务getLengthTask开始
等待getLengthTask.
5: 回到AccessTheWebAsync
任务getStringTask已经完成.
执行return语句.
退出AccessTheWebAsync.
6: 回到startButton_Click
任务getLengthTask已经完成.
AccessTheWebAsync 的结果被存储到contentLength.
显示contentLength 然后退出.