在异步编程中,如果稍有不注意,就会造成死锁问题。何为死锁:即两个以上的线程同时争夺被互相锁住的资源,两个都不放手。
在UI或asp.net中,容易造成死锁的代码如下所示:
private void btnSet_Click(object sender, EventArgs e) { Task<int> task= DelayAsync(); //在同步代码中有wait同步等待函数 task.Wait(); this.Text = "测试"; } private async Task<int> DelayAsync() { //在异步方法中,有await异步等待函数 await Task.Delay(1000); return 100; }
当执行btnSet_Click方法时,程序就会造成死锁,this.Text="测试" 这一行代码永远都不会执行。
为何会造成死锁?首先,task.Wait()方法造成了当前主线程被阻塞,其次,DelayAsync方法中的Task.Delay(1000)(由挂起到激活状态)运行结束时会捕获当前的上下文线程,而当前的上下文线程已经被阻塞,故造成死锁。
解决的方法有两个:
1、将btnSet_click变为异步方法,全部采用异步等待的方式;
private async void btnSet_Click(object sender, EventArgs e) { int result=await DelayAsync(); this.Text = result.ToString(); } private async Task<int> DelayAsync() { await Task.Delay(1000); return 100; }
2、在异步方法中使用Task.ConfigureAwait(false),不将await产生的延时任务送回到原始上下文线程中,即不会去捕获原始上下文,并在其新开辟的线程中结束其代码的运行,如下面的代码。
备注:如果是这种方式,则在异步方法的Task.ConfigureAwait(false)语句之后的代码块中,获取不了原始上下文的某些变量,容易造成NullReferenceException空值引用异常。
private async void btnSet_Click(object sender, EventArgs e) { Task<int> task = DelayAsync(); task.Wait(); this.Text = task.Result.ToString(); } private async Task<int> DelayAsync() { await Task.Delay(1000).ConfigureAwait(false); return 100; }