问题
在使用 Abp 框架的后台作业时, 当后台作业抛出异常, 会导致整个程序崩溃. 在 Abp 框架的底层执行后台作业的时候, 有 try/catch 语句块用来捕获后台任务执行时的异常, 但是在这里没有生效.
原始代码如下:
调用接口时的效果:
原因
出现这种情况是因为任何异步方法返回 void 时, 抛出的异常都会在 async void 方法启动时, 处于激活状态的同步上下文 (SynchronizationContext) 触发, 我们的所有 Task 都是放在线程池执行的.
所以在上述样例当中, 此时 AsyncVoidMethodBuilder.Create() 使用的同步上下文为 null , 这个时候 ThreadPool 就不会捕获异常给原有线程处理, 而是直接抛出.
线程池在底层使用 AsyncVoidMethodBuilder.Craete() 所拿到的同步上下文, 所捕获异常的代码如下:
虽然你可以通过挂载 AppDoamin.Current.UnhandledException 来监听异常, 不过你是没办法从异常状态恢复的.
解决
可以使用 AsyncBackgroundJob 替换掉之前的 BackgroundJob<TArgs> , 只需要实现它的 Task ExecuteAsync(TArgs args) 方法即可.
作者: myzony
来源: http://www.jianshu.com/p/9c0431f6f4ce