执行下面的程序,出现这个异常,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
版权声明:本文为博主原创文章,转载请附上博文链接!

