执行下面的程序,出现这个异常,why?

原因是: await Task.Delay(50,_token); 这句传入了_token,用户取消时会触发
_token.ThrowIfCancellationRequested();
这个时候,下面的if不会命中,而是直接向调用者抛异常。
if (_token.IsCancellationRequested)
如果写成await Task.Delay(50),则不会抛这个异常,而是直接温柔的退出task。
抛出的这个异常本身没有问题的,这样在调者那里你try处理一下,可以弹窗提示一下用户“异步已经取消”。
至于Vs断下来显示这个异常,也是没有问题的,即使你有try捕获这个异常,VS也会断下来显示这个异常。
不相信的话,你可以写句 throw new ArgumentNullException(); 这代码是人工抛了一个异常,VS也会断下来并显示这个异常。
(
貌似VS的异常设置是可以配置成命中异常后不要停止下来,勇哥忘记是哪项了。。。
2023.12.18勇哥注:找到了,按下图设置即可:
详细请参考:http://www.skcircle.com/?id=2242

)
这个只是VS的调试功能在告诉你异常被命中了,当你在正常运行执行exe程序的时候,是没有问题的。
另外,如果在一个同步方法调用异步方法,也可能出现这种问题。
演示代码:
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Threading;
using System.Threading.Tasks;
using System.Windows.Forms;
namespace 异步编程_进度条
{
public partial class Form1 : Form
{
CancellationTokenSource source;
CancellationToken token;
public Form1()
{
InitializeComponent();
progressBar1.Maximum = 100;
progressBar1.Minimum = 0;
progressBar1.Step = 1;
}
private async void button1_Click(object sender, EventArgs e)
{
try
{
source = new CancellationTokenSource();
token = source.Token;
token.Register(new Action(()=>
{
MessageBox.Show($"已经处理{progressBar1.Value}%,被用户中止!");
progressBar1.Value = 0;
}));
await updateProcess(token);
}
catch(Exception ex)
{
//MessageBox.Show(ex.Message);
}
}
private Task updateProcess(CancellationToken _token)
{
var f1 = Task.Run(async delegate ()
{
for (int i = 0; i <= 100; i++)
{
if (_token.IsCancellationRequested)
{
//_token.ThrowIfCancellationRequested();
//progressBar1.BeginInvoke((MethodInvoker)delegate { progressBar1.Value = 0; });
break;
}
progressBar1.BeginInvoke((MethodInvoker)delegate { progressBar1.PerformStep(); });
await Task.Delay(50,_token);
//Thread.Sleep(200);
}
});
return f1;
}
private void button2_Click(object sender, EventArgs e)
{
source.Cancel();
}
}
}
.
---------------------
作者:hackpig
来源:www.skcircle.com
版权声明:本文为博主原创文章,转载请附上博文链接!


少有人走的路
















