- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我有一个与 WCF 服务通信的 WPF 应用程序。我目前正在使用以下 async
从我的 ViewModels(我正在使用 MVVM 模式)调用我的 WCF 服务基于模式:
public async override void MyCommandImplementation()
{
using (var proxy = new MyProxy())
{
var something = await proxy.GetSomethingAsync();
this.MyProperty = something;
}
}
因为我遵循 MVVM 模式,所以我有 ICommand
我的 ViewModel 公开的公共(public)属性,因此关联的命令实现不会返回 Task<T>
对象,因为它们就像事件处理程序。所以异常处理实际上非常简单,即我能够使用以下模式捕获从我的 WCF 服务抛出的任何异常:
public async override void MyCommandImplementation()
{
try
{
using (var proxy = new MyProxy())
{
var something = await proxy.GetSomethingAsync();
}
}
catch (FaultException<MyFaultDetail> ex)
{
// Do something here
}
}
到目前为止一切顺利,如果服务器抛出一个异常,该异常由于自定义 WCF 行为而自动转换为 SOAP 错误,则一切都按预期进行。
因为我有一些可以在我的服务中随处抛出的常见异常(例如,每个 WCF 操作都可以抛出 AuthenticationException
,这将在客户端转换为 FaultException<AuthenticationFaultDetail>
异常),我决定在我的应用程序的一个常见位置处理一些异常,即通过处理 Application.DispatcherUnhandledException
事件。这工作得很好,我可以捕获我所有的 FaultException<AuthenticationFaultDetail>
异常无处不在,向用户显示错误消息,并阻止应用程序退出:
private static void Application_DispatcherUnhandledException(object sender, DispatcherUnhandledExceptionEventArgs e)
{
// Exception handler that handles some common exceptions
// such as FaultException<AuthenticationFaultDetail>
if (GlobalExceptionHandler.HandleException(e.Exception))
{
// Expected exception, so we're fine
e.Handled = true;
}
else
{
// We're not fine. We couldn't handle the exception
// so we'll exit the application
// Log...etc.
}
}
一切都很好,因为 FaultException
多亏了 async
被扔进了 UI 线程模式和同步上下文在 await
之前/之后切换关键字。
我的问题是,其他异常可能会在我的 UI 线程之外的另一个线程中抛出,例如在 EndPointNotFoundException
的情况下扔在 await proxy.GetSomethingAsync();
行(服务器端 WCF 服务关闭的情况)。
这些异常不会在 Application.DispatcherUnhandledException
中处理事件处理程序,因为它们不会在 UI 线程中抛出。我可以在 AppDomain.UnhandledException
中处理它们事件,但除了做一些日志记录和退出应用程序之外,我无法做任何其他事情(基本上没有“e.Handled”之类的属性)。
所以我的问题是:如果在我的应用程序的某个位置进行异步 WCF 调用,我该如何处理后台线程中抛出的异常?
我现在能想到的最好的是像下面这样的东西:
public class ExceptionHandler : IDisposable
{
public void HandleException(Exception ex)
{
// Do something clever here
}
public void Dispose()
{
// Do nothing here, I just want the 'using' syntactic sugar
}
}
...
public async override void MyCommandImplementation()
{
using (var handler = new ExceptionHandler())
{
try
{
using (var proxy = new MyProxy())
{
var something = await proxy.GetSomethingAsync();
}
}
catch (FaultException<MyFaultDetail> ex)
{
// Do something here
}
catch (Exception ex)
{
// For other exceptions in any thread
handler.HandleException(ex);
}
}
}
但这需要我重构大量代码(每次异步调用 Web 服务时)。
任何能让我不重构大量代码的想法都会有所帮助。
最佳答案
通常,我不太喜欢集中式/全局异常处理。我个人的偏好是要么在任何地方处理异常,要么编写您自己的代理包装器对象来处理/翻译预期的故障异常。
也就是说,您可以考虑一种方法(尽管它需要修改所有命令)。
首先,将实际逻辑分解为 async Task
方法,如下所示:
public async Task MyCommandAsync()
{
try
{
using (var proxy = new MyProxy())
{
var something = await proxy.GetSomethingAsync();
}
}
catch (FaultException<MyFaultDetail> ex)
{
// Do something here
}
}
public async override void MyCommandImplementation()
{
MyCommandAsync();
}
通常,我建议使用 async Task ExecuteAsync
方法和匹配 async void Execute
来实现 async ICommand
,它只会执行 等待 ExecuteAsync();
。除了 async void
方法不是 await
ing Task
之外,我上面的示例几乎相同。这很危险,我将在下面解释。
将逻辑保留在 async Task
中会给您带来一个巨大的优势:您可以更轻松地进行单元测试。此外,async Task
方法具有不同的异常处理,您可以(ab)使用它们来解决您的问题。
async Task
方法 - 如果返回的 Task
从未被 await
ed - 将引发 TaskScheduler.UnobservedTaskException
.请注意,这不会使您的进程崩溃(从 .NET 4.5 开始);您的处理程序必须决定最佳响应。由于您的async void
方法不等待
您的async Task
返回的Task
> 方法,任何异常都将在 UnobservedTaskException
中结束。
所以这会起作用,但它有一个严重的副作用:任何未观察到的 Task
异常将在同一个处理程序中结束(不仅仅是来自您的 ICommand
s).未观察到的任务异常在 .NET 4.5 中更改为默认忽略的原因是这种情况在 async
代码中不再异常。例如,考虑以下代码,它将尝试从两个不同的 url 下载并获取第一个响应:
async Task<string> GetMyStringAsync()
{
var task1 = httpClient.GetAsync(url1);
var task2 = httpClient.GetAsync(url2);
var completedTask = await Task.WhenAny(task1, task2);
return await completedTask;
}
在这种情况下,如果较慢的 url 导致错误,则该异常将发送到 UnobservedTaskException
。
关于c# - 在后台线程中全局捕获从 WCF 异步调用引发的异常,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/19158250/
WCF 服务、WCF RIA 服务和 WCF 数据服务之间有什么区别? 最佳答案 WCF 是一般服务的通信基础设施。 WCF RIA 服务自动生成客户端和服务器代理对象以方便应用程序开发,并依赖 WC
我想在我的 WPF 项目中使用 WCF 服务 (.svc)。, 我正在尝试创建一个服务。但在 Visual Studio 中,我们有“WCF 服务库”和“WCF 服务应用程序”。我两个都试了。 当我们
我正在开发 WCF Web 服务,并使用 WCF 服务应用程序模板来执行此操作。 创建“WCF 服务应用程序”是否满足此要求?与 WCF 服务应用程序相比,创建 WCF 服务库有哪些优势? 最佳答案
我是 WCF 的新手,对 Web 服务进行编码的经验有限。 在工作中,所有面向网络服务的事物都被要求使用 WCF。我需要做的工作涉及查询一个非 WCF Web 服务,该服务显然是用 Java 构建的,
我有一个数据契约(Contract)说用户。它是可序列化的并且可以通过网络传输。我想要一个操作契约(Contract) SaveUser()。我可以将 SaveUser(User user) 作为运营
我一直在开发一个使用 WCF 访问服务器端逻辑和数据库的 WPF 应用程序。 我从一个 WCF 客户端代理对象开始,我反复使用它来调用服务器上的方法。使用代理一段时间后,服务器最终会抛出异常: Sys
不要添加关于不同 WCF 堆栈的另一篇 SO 帖子,但我想在浪费更多开发时间之前确保我朝着正确的方向前进...... 我的场景 - 我们公司有许多 Web 应用程序,它们都访问同一系列的数据库。所有应
我是WCF技术的新手,我想知道RESTful WCF服务和普通WCF服务有什么区别。 RESTful 服务相对于普通 WCF 服务有哪些优势? 谢谢。 最佳答案 REST服务基于HTTP协议(prot
我正在构建的应用程序公开了多个 WCF 服务(A、B)。在内部,它消耗了在我们的内部网络(X、Y)上运行的其他几个 WCF 服务。 使用 WCF 消息日志记录,我希望仅记录我们的服务 A、B 与调用它
我们需要从另一个 WCF 服务调用 WCF 服务。为了测试这一点,我构建了一个示例控制台应用程序来显示一个简单的字符串。设置是: 控制台应用程序 -> WCF 服务 1 -> WCF 服务 2 Con
假设永远不会直接查询数据的情况。 AKA,总会有一些必须发生的过滤逻辑和/或业务逻辑。 什么时候是在 ajax/js 之外使用数据服务的好理由? 请不要访问此页面 http://msdn.micros
我在尝试将所有常规 WCF 调用转换为异步 WCF 调用时遇到问题。我发现我重构了很多代码,但不确定具体该怎么做。我使用了我找到的方法 here但遇到了我需要事情按顺序发生的问题。 private v
我在 IIS 上有一个 WCF 服务,一些 .net Web 应用程序正在使用它。我的任务是编写一个新的 WCF 服务,要求现有的 Web 应用程序可以使用新服务而无需更改它们的 web.config
我正在尝试用外部提供 WSDL 的 WCF 等效服务替换 WSE 服务。 首先,我使用 svcutil 和 wsdl 生成所有服务和客户端类(ATP,我只关心服务实现。)我生成了一个空的 WCF 服务
场景是这样的:有2个WCF Web Services,一个是客户端(WCFClient),一个是服务端(WCFServer),部署在不同的机器上。我需要他们两个之间的证书通信。 在服务器 WCF 上,
我在 Visual Studio 2013 中创建一个 WCF 服务并将其发布到 IIS。我可以在另一个项目中添加服务引用并使用该服务的方法。当我转到 IIS 服务器管理器时,我看到 WCF 激活及其
我是 .net 的新手,对 WCF 知之甚少,如果有任何愚蠢的问题,请耐心等待。我想知道如果我的代码没有显式生成任何线程,WCF 如何处理 SELF-HOST 场景中的同时调用。因此,在阅读了很多关于
我正在为应用程序开发一个面向服务的体系结构,我希望这些服务既可以通过 WCF 公开,也可以通过一个简单的库使用。理想情况下,我想减少重复代码。 从概念上讲,这映射到: Client => WCF Se
我有一个小型测试网络服务来模拟我在现实世界应用程序中注意到的一些奇怪的东西。由于演示显示与应用程序相同的行为,为了简洁起见,我将使用演示。 简而言之,我的服务接口(interface)文件如下所示(您
我首先为我的 WCF 服务启动了我的订阅者,然后继续发布我的发布者的帖子。我的订阅者能够收到帖子。 其次,我关闭了我的第一个订阅者并再次打开它以订阅相同的服务,即所谓的已订阅该服务的第二个订阅者。再一
我是一名优秀的程序员,十分优秀!