异步方法在完成其工作之前即返回到调用方法,然后在调用方法继续执行的时候完成其工作。
特点:
async Task CountCharactersAsync(int id,string site)
{
Console.WriteLine("Starting CountCharacters");
WebClient wc = new WebClient();
//await 表达式
string result = await wc.DownloadStringTaskAsync( new Uri(site) );
//Lambda 表达式
// var result = await Task.Run(() => wc.DownloadString(new Uri(site)));
return result = Length;
}
async 关键字
返回类型:
Task someTask = DoStuff.CalculateSumAsync(5,6);
...
someTask.Wait();
Task value = DoStuff.CalcuateSumAsync(5,6);
...
Console.WriteLine($"Value:{ value.Result }");
class Program
{
static void Main(string[] args)
{
ValueTask value = DoAsynStuff.CalculateSumAsync(0, 6);
//处理其他事情
Console.WriteLine($"Value:{ value.Result }");
value = DoAsynStuff.CalculateSumAsync(5, 6);
//处理其他事情
Console.WriteLine($"Value:{ value.Result }");
Console.ReadKey();
}
static class DoAsynStuff
{
public static async ValueTask CalculateSumAsync(int i1,int i2)
{
// 如i1 == 0,则可以避免执行长时间运行的任务
if(i1 == 0)
{
return i2;
}
int sum = await Task.Run(() => GetSum(i1, i2));
return sum;
}
private static int GetSum(int i1,int i2){ return i1 + i2; }
}
}
我的编程环境不支持ValueTask
void 和 Task 的区别代码例子:
class MyDownloadString
{
const string Str = "https://www.baidu.com/";
Stopwatch sw = new Stopwatch();
public async void DoRun()
{
sw.Start();
await returnTaskValue();
Console.WriteLine("----------:{0,4:N0}", sw.Elapsed.TotalMilliseconds);
returnVoidValue();
Console.WriteLine("++++++++++:{0,4:N0}", sw.Elapsed.TotalMilliseconds);
Thread.Sleep(500);
Console.WriteLine();
Console.WriteLine("暂停线程,为的是测试Task,跟 Task 和 void 的无关:");
Task t = CountCharacters("Task");
Console.WriteLine("**********:{0,4:N0}", sw.Elapsed.TotalMilliseconds);
//总结:
//1、如果 Task 和 void 异步方法里有 await,则方法内需要等待完成才能返回
//假设 Task 和 void 异步方法内有 await:
//2、在同层级方法内调用异步方法,Task 方法 添加 await 等待任务完成才能执行下一个语句
//void 方法并不会等待任务完成,而继续执行下一个语句
//3、 Task 在同一层方法被调用,则执行异步处理。跟 void 一样。
//4、同一层方法作为调用者,如果被调用者为异步方法时:
//执行 void 和 Task方法,不需要 await,则执行异步处理。
//执行 Task 方法 需要 await 执行等待任务完成,才能执行下一个语句;否则,就异步处理。
}
private async Task returnTaskValue()
{
Console.WriteLine("Task_Start:{0,4:N0}", sw.Elapsed.TotalMilliseconds);
var result = await Task.Run(() => CountCharacters("Task"));
Console.WriteLine("Task_End:{0,4:N0}", sw.Elapsed.TotalMilliseconds);
}
private async void returnVoidValue()
{
Console.WriteLine("Void_Start:{0,4:N0}", sw.Elapsed.TotalMilliseconds);
var result = await Task.Run(() => CountCharacters("Void"));
Console.WriteLine("Void_End:{0,4:N0}", sw.Elapsed.TotalMilliseconds);
}
private async Task CountCharacters(string strId)
{
WebClient wc1 = new WebClient();
Console.WriteLine("{0}_DownloadBefore:{1,4:N0}", strId, sw.Elapsed.TotalMilliseconds);
var result = await wc1.DownloadStringTaskAsync(new Uri("https://www.baidu.com/"));
Console.WriteLine("{0}_DownloadAfter:{1,4:N0}", strId, sw.Elapsed.TotalMilliseconds);
return result;
}
}
输出结果:
>Task_Start: 1
Task_DownloadBefore: 68
Task_DownloadAfter: 420
Task_End: 435
----------: 435
Void_Start: 436
Void_DownloadBefore: 437
++++++++++: 440
Void_DownloadAfter: 469
Void_End: 469
暂停线程,为的是测试Task,跟 Task 和 void 的无关:
Task_DownloadBefore: 950
**********: 952
Task_DownloadAfter: 983
void 返回类型使用“调用并忘记”的异步方法:
class Program
{
static void Main(string[] args)
{
DoAsynStuff.CalculateSumAsync(5, 6);
//如果参数值为1,则先等待void 方法执行完成之后才能执行下一个语句
//这样做法会有安全隐患,适合 void 异步方法处理时间短,Sleep 参数值设置足够大
Thread.Sleep(200);
Console.WriteLine("Program Exiting");
Console.ReadKey();
}
static class DoAsynStuff
{
public static async void CalculateSumAsync(int i1,int i2)
{
int value = await Task.Run(() => GetSum(i1, i2));
Console.WriteLine("Value:{0}", value);
}
private static int GetSum(int i1,int i2)
{
return i1 + i2;
}
}
}
异步方法的结构包含三个不同区域:
图-阐明了一个异步方法的控制流
目前有两个控制流: 一个在异步方法内,一个在调用方法内。
当后续部分遇到 retun 语句或到达方法末尾时:
await 表达式指定了一个异步执行的任务。
await task //task:空闲对象
一个空闲对象即是一个 awaitable 类型的实例。 awaitable 类型是指包含 GetAwaiter 方法的类型,该方法没有参数,返回一个 awaiter 类型的对象。
awaiter 类型包含以下成员:
bool IsCompleted{ get; }
void OnCompleted(Action);
void GetResult();
T GetResult();
使用 Task.Run 方法来创建一个 Task。
//是以 Func 委托为参数
public static Task Run(Func function);
//还有其他重载的类型
...
创建委托三种实现方式:
class MyClass
{
public int Get10()
{
return 10;
}
public async Task DoWorkAsync()
{
Func ten = new Func(Get10);
int a = await Task.Run(ten);
int b = await Task.Run(new Func(Get10));
int c = await Task.Run(()=> { return 10; });
Console.WriteLine($"{ a } { b } { c }");
}
}
class Program
{
static void Main(string[] args)
{
Task t = (new MyClass()).DoWorkAsync();
t.Wait();
Console.ReadKey();
}
}
输出结果:
10 10 10
表-Task.Run 重载的返回类型和签名
返回类型 | 签名 |
---|---|
Task | Run( Action action) |
Task | Run( Action action, CancellationToken token ) |
Task |
Run( Func |
Task |
Run( Func |
Task | Run( Func |
Task | Run( Func |
Task |
Run( Func |
Task |
Run( Func |
表-可作为 Task.Run 方法第一个参数的委托类型
委托类型 | 签名 | 含义 |
---|---|---|
Action | void Action() | 不需要参数且无返回值的方法 |
Func |
TResult Func() | 不需要参数,但返回 TRsult 类型对象的方法 |
Func |
Task Func() | 不需要参数,但返回简单 Task 对象的方法 |
Func |
Task |
不需要参数,但返回 Task |
4个 await 语句的示例:
static class MyClass
{
public static async Task DoWorkAsync()
{
//Run(Action)
await Task.Run(() => Console.WriteLine(5.ToString()));
//Run(TResult Func())
Console.WriteLine((await Task.Run(() => 6)).ToString());
//Run(Task Func())
await Task.Run(() => Task.Run(() => Console.WriteLine(7.ToString())));
//Run(Task Func())
int value = await Task.Run(() => Task.Run(() => 8));
Console.WriteLine(value.ToString());
}
}
class Program
{
static void Main(string[] args)
{
Task t = MyClass.DoWorkAsync();
t.Wait();
Console.WriteLine("Press Enter key to exit");
Console.ReadKey();
}
}
Lambda 函数() = > GetSum(5,6) 满足 Func
int value = awaite Task.Run(() => GetSum(5,6));
一些 .NET 异步方法允许你请求终止执行。
System.Threading.Tasks 命名空间中有两个类是为此目的而设计的:CancellationToken 和 CancellationTokenSource。
使用这两个类,触发取消行为:
class Program
{
static void Main(string[] args)
{
CancellationTokenSource cts = new CancellationTokenSource();
CancellationToken token = cts.Token;
MyClass mc = new MyClass();
Task t = mc.RunAsync(token);
//Thread.Sleep(3000);
//cts.Cancel();
t.Wait();
Console.WriteLine($"Was Cancelled:{ token.IsCancellationRequested }");
Console.ReadKey();
}
class MyClass
{
public async Task RunAsync(CancellationToken ct)
{
if (ct.IsCancellationRequested)
return;
await Task.Run(() => CycleMethod(ct), ct);
}
void CycleMethod(CancellationToken ct)
{
Console.WriteLine("Starting CycleMethod");
const int max = 5;
for(int i = 0; i < max;i++)
{
if (ct.IsCancellationRequested)
return;
Thread.Sleep(1000);
Console.WriteLine($"{ i + 1 } of { max } iterations completed");
}
}
}
}
输出结果:
Starting CycleMethod
1 of 5 iterations completed
2 of 5 iterations completed
3 of 5 iterations completed
4 of 5 iterations completed
5 of 5 iterations completed
Was Cancelled:False
若在 Main 方法里恢复执行这两行代码:
Thread.Sleep(3000);//3秒后取消任务执行
cts.Cancel();
输出结果:
Starting CycleMethod
1 of 5 iterations completed
2 of 5 iterations completed
3 of 5 iterations completed
Was Cancelled:True
你的代码会继续执行其他任务,但在某个点上可能会需要等待某个特殊 Task 对象完成,然后再继续。
static class MyDownloadString
{
public static void DoRun()
{
Task t = CountCharactersAsync("http://illustratedcsharp.com");
t.Wait();//等待任务 t 结束
Console.WriteLine($"The task has finished,returning value { t.Result }.");
}
private static async Task CountCharactersAsync(string site)
{
string result = await new WebClient().DownloadStringTaskAsync(new Uri(site));
return result.Length;
}
}
class Program
{
static void Main(string[] args)
{
MyDownloadString.DoRun();
Console.ReadKey();
}
}
输出结果:
The task has finished,returning value 5164.
对于一组 Task ,可以等待所有任务都结束,也可以等待某一个任务结束。
class MyDownloadString
{
Stopwatch sw = new Stopwatch();
public void DoRun()
{
sw.Start();
Task t1 = CountCharactersAsync(1,"http://illustratedcsharp.com");
Task t2 = CountCharactersAsync(2, "http://illustratedcsharp.com");
//Task.WaitAll(t1, t2);
//Task.WaitAny(t1, t2);
Console.WriteLine("Task 1: {0} Finished",t1.IsCompleted?"":"Not");
Console.WriteLine("Task 2: {0} Finished", t2.IsCompleted ? "" : "Not");
Console.Read();
}
private async Task CountCharactersAsync(int id,string site)
{
WebClient wc = new WebClient();
string result = await wc.DownloadStringTaskAsync(new Uri(site));
Console.WriteLine(" Call {0} completed: {1,4:N0} ms", id, sw.Elapsed.TotalMilliseconds);
return result.Length;
}
}
class Program
{
static void Main(string[] args)
{
MyDownloadString ds = new MyDownloadString();
ds.DoRun();
Console.ReadKey();
}
}
输出结果:
不使用 WaitAll 和 WaitAny 方法时,t1 和 t2 异步执行。
Task 1: Not Finished
Task 2: Not Finished
Call 1 completed: 598 ms
Call 2 completed: 600 ms
1)如果在 DoRun 方法里,恢复 Task.WaitAll(t1, t2); 代码:
Task.WaitAll(t1, t2);
//Task.WaitAny(t1, t2);
输出结果:
使用了 WaitAll 方法后,所有任务都要等待完成,之后代码才能继续执行。
Call 2 completed: 695 ms
Call 1 completed: 617 ms
Task 1: Finished
Task 2: Finished
2)如果在 DoRun 方法里,恢复 Task.WaitAny(t1, t2); 代码:
//Task.WaitAll(t1, t2);
Task.WaitAny(t1, t2);
输出结果:
使用了 WaitAll 方法后,至少有一个 Task 需要等待完成才能继续执行。
Call 2 completed: 659 ms
Call 1 completed: 658 ms
Task 1: Not Finished
Task 2: Finished
签名 | 描述 |
---|---|
WaitAll | – |
void WaitAll(params Task[] tasks) | 等待所有任务完成 |
bool WaitAll(Task[] tasks, int millisecondsTimeout) | 等待所有任务完成。如果在超时时限内没有全部完成,则返回 false 并继续执行 |
void WaitAll(Task[] tasks, CancellationToken token) | 等待所有任务完成,或等待 CancellationToken 发出取消信号 |
bool WaitAll(Task[] tasks, TimeSpan span) | 等待所有任务完成。如果在超时时限内没有全部完成,则返回 false 并继续执行 |
bool WaitAll(Task[] tasks, int millisecondsTimeout, CancellationToken token) | 等待所有任务完成,或等待 CancellationToken 发出取消信号。如果在超时时限内没有发生上述情况,则返回 false 并继续执行 |
WaitAny | – |
void WaitAny(params Task[] tasks) | 等待任意一个任务完成 |
bool WaitAny(Task[] tasks, int millisecondsTimeout) | 等待任意一个任务完成。如果在超时时限内没有完成的,则返回 false 并继续执行 |
void WaitAny(Task[] tasks, CancellationToken token) | 等待任意一个任务完成,或等待 CancellationToken 发出取消信号 |
bool WaitAny(Task[] tasks, TimeSpan span) | 等待任意一个任务完成。如果在超时时限内没有完成的,则返回 false |
bool WaitAny(Task[] tasks, int millisecondsTimeout, CancellationToken token) | 等待任意一个任务完成,或等待 CancellationToken 发出取消信号。如果在超时时限内没有发生上述情况,则返回 false 并继续执行 |
可以通过 Task.WhenAll 和 Task.WhenAny 方法,来实现在异步方法中等待一个或所有任务完成。这两个方法称为组合子。
class MyDownloadString
{
Stopwatch sw = new Stopwatch();
public void DoRun()
{
sw.Start();
Task t = CountCharactersAsync("http://www.baidu.com", "http://illustratedcsharp.com");
Console.WriteLine("DoRun: Task {0}Finshed", t.IsCompleted ?"":"Not");
Console.WriteLine("DoRun: Result = {0}", t.Result);
}
private async Task CountCharactersAsync(string site1,string site2)
{
WebClient wc1 = new WebClient();
WebClient wc2 = new WebClient();
Task t1 = wc1.DownloadStringTaskAsync(new Uri(site1));
Task t2 = wc2.DownloadStringTaskAsync(new Uri(site2));
List> tasks = new List>();
tasks.Add(t1);
tasks.Add(t2);
await Task.WhenAll(tasks);//关键代码语句
Console.WriteLine(" CCA: T1 {0}Finished", t1.IsCompleted ? "" : "Not");
Console.WriteLine(" CCA: T2 {0}Finished", t2.IsCompleted ? "" : "Not");
return t1.IsCompleted ? t1.Result.Length : t2.Result.Length;
}
}
class Program
{
static void Main(string[] args)
{
MyDownloadString ds = new MyDownloadString();
ds.DoRun();
Console.ReadKey();
}
}
输出结果:
在异步方法内,需要等待所有任务完成后,之后代码才能继续执行。
DoRun: Task NotFinshed
CCA: T1 Finished
CCA: T2 Finished
DoRun: Result = 9269
若把 Task.WhenAll 方法 改为 Task.WhenAny 方法:
输出结果:
在异步方法内,至少有一个任务需要等待完成,之后代码才能继续执行。
DoRun: Task NotFinshed
CCA: T1 Finished
CCA: T2 NotFinished
DoRun: Result = 9269
Thread.Sleep 会阻塞线程,而 Task.Delay 不会阻塞线程,线程还可以继续处理其他工作。
class Simple
{
Stopwatch sw = new Stopwatch();
public void DoRun()
{
Console.WriteLine("Caller:Before call");
ShowDelayAsync();
Console.WriteLine("Caller:After call");
}
private async void ShowDelayAsync()
{
sw.Start();
Console.WriteLine($" Before Delay:{ sw.ElapsedMilliseconds }");
await Task.Delay(1000);
Console.WriteLine($" After Delay : { sw.ElapsedMilliseconds }");
}
}
class Program
{
static void Main(string[] args)
{
Simple ds = new Simple();
ds.DoRun();
Console.ReadKey();
}
}
输出结果:
Caller:Before call
Before Delay:0
Caller:After call
After Delay : 1052
Delay 方法包含4个重载,允许以不同方法来指定时间周期,同时还允许使用 CancellationToken 对象。
签名 | 描述 |
---|---|
Task Delay(int millisecondsDelay) | 在以毫秒表示的延迟时间到期后,返回完成的 Task 对象 |
Task Delay(TimeSpan delay) | 在以 .NET TimeSpan 对象表示的延迟时间到期后,返回完成的 Task 对象 |
Task Delay(int millisecondsDelay, CancellationToken token) | 在以毫秒表示的延迟时间到期后,返回完成的 Task 对象。可通过取消令牌来取消该操作 |
Task Delay(TimeSpan delay, CancellationToken token) | 在以 .NET TimeSpan 对象表示的延迟时间到期后,返回完成的 Task 对象。可通过取消令牌来取消该操作 |