记一次Task抛异常,调用线程处理而引发的一些随想
记一次Task抛异常,调用线程处理而引发的一些随想
多线程调用,任务线程抛出异常如何在另一个线程(调用线程)中捕获并进行处理的问题。
1.任务线程在任务线程执行语句上抛出异常。
例如:
private void button2_Click(object sender, EventArgs e)
{
try
{
var task = Task.Factory.StartNew<bool>(() =>
{
//Do Some Things
throw new Exception("Task Throw Exception!");
//return true;
}); //var result = task.Wait(20000);
var result = task.Result;
}
catch (Exception ex)
{ } }
调试结果:在Task.Rrsult或者Wait时可以抛出任务异常,并在调用线程中通过try-catch捕获处理。

2.任务线程在异步委托执行语句上抛出异常。
private void button3_Click(object sender, EventArgs e)
{
var fun = new Func<int>(() =>
{
//do sonmething
throw new Exception("Task Throw Exception!");
return ;
});
try
{
var task = Task.Factory.StartNew<bool>(() =>
{
try
{
var res = fun.BeginInvoke(null, null);
//do some thing
var ob = fun.EndInvoke(res);
}
catch (Exception ex)
{ throw ex;
}
return true;
});
var result = task.Wait();
//var result1 = task.Result;
}
catch (Exception ex)
{ }
}
调试可知:异步委托在调用EndInvoke(res)获取结果时可以捕获委托内部异常并抛出由外部Task抓取。
2.任务线程在窗口句柄(创建控件)线程上抛异常现象。
control.invoke(参数delegate)方法:在拥有此控件的基础窗口句柄的线程上执行指定的委托。
control.begininvoke(参数delegate)方法:在创建控件的基础句柄所在线程上异步执行指定委托。
即invoke表是同步、begininvoke表示异步。但是如何来进行同步和异步呢?
2.1Invoke方法执行规则
Invoke的原理是借助消息循环通知主线程,并且在主线程执行委托。直接代码查看:
private void button1_Click(object sender, EventArgs e)
{
//Invoke的原理是借助消息循环通知主线程,并且在主线程执行委托。
try
{
var thIdMain = Thread.CurrentThread.ManagedThreadId;
Console.WriteLine($"Load start: Main Thread ID:{thIdMain}");
var task = Task.Factory.StartNew<bool>(() =>
{
var taskId = Thread.CurrentThread.ManagedThreadId;
Console.WriteLine($"Task start: Task Thread ID:{taskId}");
var res = this.Invoke(new Func<int>(() =>
{
var InvokeId = Thread.CurrentThread.ManagedThreadId;
Console.WriteLine($"Invoke in: Begion Invoke Thread ID:{InvokeId}");
//do sonmething
return ;
}));
taskId = Thread.CurrentThread.ManagedThreadId;
Console.WriteLine($"Invoke out ,Thread ID:{taskId}");
return true; }); thIdMain = Thread.CurrentThread.ManagedThreadId;
Console.WriteLine($"Wait: Main Thread ID:{thIdMain}");
var CanLoad = task.Wait();//.Result;
thIdMain = Thread.CurrentThread.ManagedThreadId;
Console.WriteLine($"End: Main Thread ID:{thIdMain}");
}
catch (Exception) { }
}
执行输出:
Load start: Main Thread ID:1
Wait: Main Thread ID:1
Task start: Task Thread ID:3
End: Main Thread ID:1
Invoke in: Begion Invoke Thread ID:1
Invoke out ,Thread ID:3
查看输出顺序说明:invoke在主线程中执行,但是,invoke后面的代码必须在Invoke委托方法执行完成后,才能继续执行;而invoke在主线程中执行,所以其执行时机无法确定,得等消息循环(主线程)中其它消息执行后才能进行。
2.2BeginInvoke方法执行规则
BeginInvoke也在主线程执行相应委托。直接代码查看:
private void button1_Click(object sender, EventArgs e)
{
//BeginInvoke
try
{
var thIdMain = Thread.CurrentThread.ManagedThreadId;
Console.WriteLine($"Load start: Main Thread ID:{thIdMain}");
var task = Task.Factory.StartNew<bool>(() =>
{
var taskId = Thread.CurrentThread.ManagedThreadId;
Console.WriteLine($"Task start: Task Thread ID:{taskId}");
var res = this.BeginInvoke(new Func<int>(() =>
{
var BegionInvokeId = Thread.CurrentThread.ManagedThreadId;
Console.WriteLine($"BeginInvoke in: Begion Invoke Thread ID:{BegionInvokeId}");
//do sonmething
return ;
}));
taskId = Thread.CurrentThread.ManagedThreadId;
Console.WriteLine($"BeginInvoke is Completed: {res.IsCompleted}, Thread ID:{taskId}");
var ob = this.EndInvoke(res);
taskId = Thread.CurrentThread.ManagedThreadId;
Console.WriteLine($"BeginInvoke out ,Thread ID:{taskId}");
// Console.WriteLine(ob.ToString());
return true;
});
long i = ;
while (i < )//延时
{
i++;
}
thIdMain = Thread.CurrentThread.ManagedThreadId;
Console.WriteLine($"Wait: Main Thread ID:{thIdMain}");
//var CanLoad = task.Wait(2000);//.Result;
thIdMain = Thread.CurrentThread.ManagedThreadId;
Console.WriteLine($"End: Main Thread ID:{thIdMain}");
//Console.WriteLine(CanLoad);
}
catch (Exception) { }
}
执行输出:
Load start: Main Thread ID:1
Task start: Task Thread ID:3
BeginInvoke is Completed: False, Thread ID:3
Wait: Main Thread ID:1
End: Main Thread ID:1
BeginInvoke in: Begion Invoke Thread ID:1
BeginInvoke out ,Thread ID:3
根据输出结果可知begininvoke所提交的委托方法也是在主线程中执行,BeginInvoke is Completed: False, Thread ID:3与Wait: Main Thread ID:1两段比较,会发现begininvoke提交委托方法后,子线程继续执行,不需要等待委托方法的完成。
总结:invoke和begininvoke都是在主线程中执行。invoke提交的委托方法执行完成后,才能继续执行;begininvoke提交委托方法后,子线程继续执行。invoke(同步)和begininvoke(异步)的含义,是相对于子线程而言的,实际上对于控件的调用总是由主线程来执行。
2.3 Control.BeginInvoke或者Control.Invoke执行委托时抛出异常
Control.Invoke执行委托时抛出异常:
private void button2_Click(object sender, EventArgs e)
{
try
{
var task = Task.Factory.StartNew<bool>(() =>
{
try
{
//Do Some Things
var res = this.Invoke(new Func<int>(() =>
{
//do sonmething
throw new Exception("Task Throw Exception!");
return ;
}));
}
catch (Exception ex)
{ throw ex;
}
return true;
});
}
catch (Exception ex)
{ }
}
执行结果:只有task中的try可以捕捉,继续抛出,主线程捕捉不到

原因分析:button2_Click方法和task中invoke都是在主线程中执行。但是,invoke必须等主线程中其它消息执行完即button2_Click代码执行完退出才有机会执行。此时button2_Click方法执行完,所分配的内存空间被回收(失效),故即便task继续抛异常均不能捕获到。而Invoke在执行完成时,task后续代码阻断并等待其执行完,后续执行代码与其在内存上属于同一 堆栈,故可以捕获到Invoke抛出的异常。
Control.BeginInvoke执行委托时抛出异常:
private void button2_Click(object sender, EventArgs e)
{
try
{
var task = Task.Factory.StartNew<bool>(() =>
{
try
{
//Do Some Things
var res = this.BeginInvoke(new Func<int>(() =>
{
//do sonmething
throw new Exception("Task Throw Exception!");
return ;
})); var ob = this.EndInvoke(res);
}
catch (Exception ex)
{ throw ex;
}
return true;
});
}
catch (Exception ex)
{ } }
执行结果:均无法捕捉异常。但是Main函数中可以。

原因分析:button2_Click方法和task中BeginInvoke都是在主线程中执行。但是,BeginInvoke须等主线程中其它消息执行完即button2_Click代码执行完退出才有机会执行。此时button2_Click方法执行完,所分配的内存空间被回收(失效),故即便task继续跑异常均不能捕获到。而BeginInvoke在执行完成时,task后续代码无须阻断等待其执行完,二者在内存上不属于同一 堆栈, 而异步调用时,异步执行期间产生的异常由CRL库捕获,你并一般在调用EndInvoke函数获取执行结果时CRL会抛出引发异步执行期间产生的异常,但是,CRL对Control.BeginInvoke特殊处理并未抛出(个人猜想,待验证)。故此时Task无法捕获到BeginInvoke抛出的异常。
一般BeginInvoke与Invoke主要用于更新控件相关属性值,特意抛异常的可能性应该比较小,如果有异常可以在该委托里面就进行解决了。此处仅作对技术研究的一个记录。
记一次Task抛异常,调用线程处理而引发的一些随想的更多相关文章
- WPF异常“调用线程无法访问此对象,因为另一个线程拥有该对象 ”
WPF中在对界面进行操作的时候,可能会遇到"调用线程无法访问此对象,因为另一个线程拥有该对象"异常,这是因为WPF中只有UI线程才能操作UI元素,非UI线程要访问UI时就会报异常了 ...
- 线程执行synchronized同步代码块时再次重入该锁过程中抛异常,是否会释放锁
一个线程执行synchronized同步代码时,再次重入该锁过程中,如果抛出异常,会释放锁吗? 如果锁的计数器为1,抛出异常,会直接释放锁: 那如果锁的计数器为2,抛出异常,会直接释放锁吗? 来简单测 ...
- ABP在领域事件中异步调用方法抛异常
在领域事件中调用UserRegistrationManager.RegisterAsync抛异常 Call UserRegistrationManager.RegisterAsync() throw ...
- CloudStack的VO在调用setRemoved方法抛异常的原因
今天在开发中发现一个问题,本来想对一个VO对象的removed值赋值,然后去update一下这条记录,一个最简单的set方法,但是在调用时直接抛异常了. 1: public void setRemov ...
- C# 线程抛异常
异常抛出 异常抛出要在线程代码中抛出,否则捕获不到 using System; using System.Threading; namespace testthread_keyword_lock { ...
- 使用Task简化Silverlight调用Wcf
原文http://www.cnblogs.com/lemontea/archive/2012/12/09/2810549.html 从.Net4.0开始,.Net提供了一个Task类来封装一个异步操作 ...
- 01 语言基础+高级:1-7 异常与多线程_day05 【异常、线程】
day05 [异常.线程] 主要内容 异常.线程 教学目标 能够辨别程序中异常和错误的区别 说出异常的分类 说出虚拟机处理异常的方式 列举出常见的三个运行期异常 能够使用try...catch关键字处 ...
- 288 day05_异常,线程
day05 [异常.线程] 主要内容 异常.线程 教学目标 [ ] 能够辨别程序中异常和错误的区别 [ ] 说出异常的分类 [ ] 说出虚拟机处理异常的方式 [ ] 列举出常见的三个运行期异常 [ ] ...
- poco json 中文字符,抛异常JSON Exception -->iconv 转换 备忘录。
起因 最近linux服务器通信需要用到json. jsoncpp比较出名,但poco 1.5版本以后已经带有json库,所以决定使用poco::json(linux 上已经用到了poco这一套框架). ...
随机推荐
- psql的时间类型,通过时间查询
psql的时间类型,通过时间查询 psql有date/timestamp类型,date只显示年月日1999-01-08,而timestamp显示年月日时分秒 1999-01-08 09:54:03.2 ...
- SpringBoot突报java.lang.NoSuchFieldError分析
SpringBoot项目,引了一个内部的工具包,竟然导致启动失败,报找不到freemarker Configuration类的一个属性,网上的解法都大同小异,最终用了自己的办法解决,花点时间记录下来, ...
- SpringBoot图文教程15—项目异常怎么办?「跳转404错误页面」「全局异常捕获」
有天上飞的概念,就要有落地的实现 概念十遍不如代码一遍,朋友,希望你把文中所有的代码案例都敲一遍 先赞后看,养成习惯 SpringBoot 图文教程系列文章目录 SpringBoot图文教程1-Spr ...
- TARS基金会:构建微服务开源生态
导语 在20世纪60至70年代,软件开发人员通常在大型机和小型机上使用单体架构进行软件开发,没有一个应用程序能够满足大多数最终用户的需求.垂直行业使用的软件代码量更小,与其他应用程序的接口更简单,而可 ...
- 软件基础1Word文档编辑
word文档编辑 启动Word2010 创建文档,<你好word>. 编辑文字. 保存的三种方式. ctrl+s. 点击文件选择保存,或另存为. 快速工具栏保存按钮. 设置字体 1.通过工 ...
- Js中的window.parent ,window.top,window.self 了解
在应用有frameset或者iframe的页面时,parent是父窗口,top是最顶级父窗口(有的窗口中套了好几层frameset或者iframe),self是当前窗口, opener是用open方法 ...
- 普通索引和唯一索引如何选择(谈谈change buffer)
假设有一张市民表(本篇只需要用其中的name和id_card字段,有兴趣的可以翻看“索引”篇,里面有建表语句) 每个人都有一个唯一的身份证号,且业务代码已经保证不会重复. 由于业务需求,市民需要按身份 ...
- php7连接mysql8
最近因为剁手买了mac所以在mac上搭建lnmp环境. 刚好看到mysql从5.7跳到8,性能据说快上一倍,果断尝鲜! lnmp基本都弄好了,但是到用php连接Mysql这一步出了岔子. 出错原因: ...
- 学习webpack基础笔记01
学习webpack基础笔记 1.webpack搭建环境最重要的就是如何使用loader和plugins,使用yarn/npm安装插件.预处理器,正确的配置好去使用 2.从0配置webpack - 1. ...
- C# RSACryptoServiceProvider 加密解密 RSA 加密解密
什么是RSA:RSA公开密钥密码体制.所谓的公开密钥密码体制就是使用不同的加密密钥与解密密钥,是一种“由已知加密密钥 推导出 解密密钥在计算上是不可行的”密码体制. 下附代码,在控制台中粘贴在启动类即 ...