单层应用程序可以通过以下方式区分异常:
Exception ex;
if (ex is System.DirectoryServices.AccountManagement.PasswordException)
...
其中ex只是一个普通的例外。
当你转移到WCF进行多层时,你将失去所有这一切,你需要使用FaultException机制。
问题在于我找不到任何方法来做到这一点。
在我的客户端中,我想捕获FaultException类型,然后区分它们,例如:
catch (FaultException ex)
{
if FaultException is (PasswordExceptionFault)
...
etc
}
有办法做到这一点吗?
否则我必须有许多catch构造 - 每种类型的FaultException都有一个。
答案 0 :(得分:8)
使用WCF服务时,必须使用FaulException,因为它是处理错误的本机Soap方法(异常也不可序列化)。
在典型情况下,您将首先在每个操作合同上添加类型化的FaultContract。
// NOTE: This is the std wcf template
[ServiceContract]
public interface IService1
{
[FaultContract(typeof(int))]
[FaultContract(typeof(string))]
[FaultContract(typeof(DateTime))]
[OperationContract]
string GetData(int value);
}
您的服务将针对您决定客户端需要故障信息的所有故障情况发送类型化的faultexception。
您的客户端可以捕获表示操作协定中指定的自定义SOAP错误的FaultException类型。
ServiceReference1.Service1Client proxy = new ServiceReference1.Service1Client();
try
{
Console.WriteLine("Returned: {0}", proxy.GetData(-5));
}
catch (FaultException<int> faultOfInt)
{
//TODO
proxy.Abort();
}
catch (FaultException<string> faultOfString)
{
//TODO
proxy.Abort();
}
catch (FaultException<DateTime> faultOfDateTime)
{
//TODO
proxy.Abort();
}
catch (FaultException faultEx)
{
Console.WriteLine("An unknown exception was received. "
+ faultEx.Message
+ faultEx.StackTrace
);
proxy.Abort();
}
catch (Exception e)
{
//generic method
Type exceptionType = e.GetType();
if (exceptionType.IsGenericType && exceptionType.GetGenericTypeDefinition() == typeof(FaultException<>))
{
PropertyInfo prop = exceptionType.GetProperty("Detail");
object propValue = prop.GetValue(e, null);
Console.WriteLine("Detail: {0}", propValue);
}
else
{
Console.WriteLine("{0}: {1}", exceptionType, e.Message);
}
}
最后,因为FaultException继承了Exception,你仍然可以使用反射来获取内部故障类型和细节,如下所示。
另请注意,WCF客户端上的通信方法的常见预期异常包括TimeoutException,CommunicationException和CommunicationException的任何派生类(如FaultException)。这些指示通信中的问题,可以通过中止WCF客户端并报告通信故障来安全地处理该问题。
答案 1 :(得分:-2)
只需使用:
if (error is FaultException<ServerTooBusyException>)
{
// Do something
}