解析Silverlight调用WCF/Rest异常的解决方法

栏目: ASP.NET · 发布时间: 6年前

内容简介:解析Silverlight调用WCF/Rest异常的解决方法

[ServiceContract]

public interface IService1

{

[OperationContract]

string GetData(int value);

}

接着新建一个服务实现类:

public class Service1 : IService1

{

public string GetData(int value)

{

int i = 0;

int j = 5 / i;

return string.Format("You entered: {0}", value);

}

}

在这里让Service1 抛出”divided by zero exception:”

<system.serviceModel>

<behaviors>

<serviceBehaviors>

<behavior name="ServiceBehavior">

<serviceDebug includeExceptionDetailInFaults="true" />

<serviceMetadata httpGetEnabled="true" />

</behavior>

</serviceBehaviors>

</behaviors>

<services>

<service behaviorConfiguration="ServiceBehavior" name="WcfService1.Service1">

</service>

</services>

</system.serviceModel>

在这里注意<serviceDebug includeExceptionDetailInFaults="true" />

在Silverlight 客户端添加服务引用,名称为:ServiceReference1.

在页面上添加一个按钮,按钮的Click事件代码如下:

private void Button_Click(object sender, RoutedEventArgs e)

{

Service1Client client = new ServiceReference1.Service1Client();

client.GetDataCompleted += new EventHandler<GetDataCompletedEventArgs>(client_GetDataCompleted);

client.GetDataAsync(35); //Try GetData

}

void client_GetDataCompleted(object sender, ServiceReference1.GetDataCompletedEventArgs e)

{

}

运行,结果如下:

解析Silverlight调用WCF/Rest异常的解决方法

可以看到实际的异常是“尝试除以0”,但是由于浏览器限制,所有的异常都是NotFound。

在msdn上有两种方法可以解决这个问题,

最简单的就是在App.xaml.cs代码里面使用RegisterPrefix来 使用备用客户端 HTTP 堆栈

public App()

{

bool bRegisterPrefix = WebRequest.RegisterPrefix(http://localhost:9541/,

WebRequestCreator.ClientHttp);

//other Code

}

再次运行代码: 解析Silverlight调用WCF/Rest异常的解决方法

这是SL调用WCF服务如何处理异常的方式,那么调用Rest服务呢?

首先要修改Web.config 节点下的serviceModel以让它支持Rest。

<system.serviceModel>

<behaviors>

<endpointBehaviors>

<behavior name="EndpointBehavior">

<webHttp helpEnabled="true" defaultOutgoingResponseFormat="Json"

faultExceptionEnabled="true" />

</behavior>

</endpointBehaviors>

<serviceBehaviors>

<behavior name="ServiceBehavior">

<serviceDebug includeExceptionDetailInFaults="true" />

<serviceMetadata httpGetEnabled="true" />

</behavior>

</serviceBehaviors>

</behaviors>

<services>

<service behaviorConfiguration="ServiceBehavior" name="WcfService1.Service1">

<endpoint behaviorConfiguration="EndpointBehavior" binding="webHttpBinding"

bindingConfiguration="" name="Rest" contract="WcfService1.IService1" />

</service>

</services>

</system.serviceModel>

在这里要设置webHttp 节点的faultExceptionEnabled=true.并且设置serviceDebug 的includeExceptionDetailInFaults 为true。

OK,服务的Web.config文件已经配置完毕了,接下来要为GetData方法添加WebGet特性修饰了。

public class Service1 : IService1

{

[WebGet()]

public string GetData(int value)

{

int i = 0;

int j = 5 / i;

return string.Format("You entered: {0}", value);

}

}

运行:

地址为:http://localhost:9541/Service1.svc/help

解析Silverlight调用WCF/Rest异常的解决方法

接着输入地址:http://localhost:9541/Service1.svc/GetData?value=3

解析Silverlight调用WCF/Rest异常的解决方法

可以看到得到了异常信息了。

注意:别忘记了添加跨域和授权文件:crossdomain.xml 和 clientaccesspolicy.xml 到网站根目录。

同样,修改SL客户端页面,添加一个Button,button的代码事件为:

private void btnRest_Click(object sender, RoutedEventArgs e)

{

WebClient wc = new WebClient();

wc.DownloadStringCompleted += new DownloadStringCompletedEventHandler(

wc_DownloadStringCompleted);

wc.DownloadStringAsync(new Uri("http://localhost:9541/Service1.svc/GetData?value=3"));

}

void wc_DownloadStringCompleted(object sender, DownloadStringCompletedEventArgs e)

{

if (e.Error != null)

{

throw e.Error;

}

}

运行,点击btnRest

解析Silverlight调用WCF/Rest异常的解决方法

可以看到,Rest 调用的结果仍然是NotFound。

提示让我们查看Response属性和Status属性。

就看看Respone属性的ResponseStrem是什么吧。

解析Silverlight调用WCF/Rest异常的解决方法

可以看到errorMessage 就是返回的错误,很明显,我们需要对它反序列化成Exception的对象。

首先尝试使用DataContractSerializer来反序列化为FaultException类

解析Silverlight调用WCF/Rest异常的解决方法

因为我们尝试反序列化为FaultException类,但是XML数据的Element名称为Fault。所以失败,难道是有Fault类 ?可是找了很久也没发现Fault类。

但是在ReadObject方法中发现了一个verifyObjectName的重载。

将代码修改为:

DataContractSerializer serializer = new DataContractSerializer(

typeof(FaultException));

//object deserializerObject = serializer.ReadObject(errorStream);

object deserializerObject = serializer.ReadObject(XmlReader.Create(errorStream),false);

重新运行:

解析Silverlight调用WCF/Rest异常的解决方法

可以发现虽然序列化是成功的,但是序列化后的值全部是错误的。

最后没办法既然有XML的异常数据,那么可以尝试解析xml数据并使用自定义异常。

首先新建SLFaultException 类,继承Exception:代码如下:

public class SLFaultException : Exception

{

public ExceptionDetail Detail { get; set; }

public SLFaultException() { }

public SLFaultException(string message) : base(message) { }

public SLFaultException(string message, ExceptionDetail detail)

: base(message)

{

Detail = detail;

}

}

完整的代码如下:

void wc_DownloadStringCompleted(object sender, DownloadStringCompletedEventArgs e)

{

if (e.Error != null)

{

if (e.Error is WebException)

{

WebResponse errorResponse = ((WebException)e.Error).Response;

Stream errorStream = errorResponse.GetResponseStream();

XElement rootElement = XElement.Load(errorStream);

XElement detailElement = rootElement

.Descendants()

.First(el => el.Name.LocalName == "ExceptionDetail");

DataContractSerializer serializer = new DataContractSerializer(

typeof(ExceptionDetail));

ExceptionDetail exceptionDetail = (ExceptionDetail)serializer.ReadObject(

detailElement.CreateReader(), true);

SLFaultException faultException = new SLFaultException(

exceptionDetail.Message, exceptionDetail);

throw faultException;

}

}

}

虽然序列化为FaultException是失败的,但是xml节点的ExceptionDetail是可以被反序列回来的,当然上面的处理WebException的过程是可以被封装的,读者自己尝试下吧,呵呵。

结果如下图:

解析Silverlight调用WCF/Rest异常的解决方法


以上所述就是小编给大家介绍的《解析Silverlight调用WCF/Rest异常的解决方法》,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对 码农网 的支持!

查看所有标签

猜你喜欢:

本站部分资源来源于网络,本站转载出于传递更多信息之目的,版权归原作者或者来源机构所有,如转载稿涉及版权问题,请联系我们

智能

智能

[法]弗雷德里克·马特尔 / 君瑞图、左玉冰 / 商务印书馆 / 2015-8 / 59.90

揭示数字化主流大趋势的最新权威论著 《主流》作者的最新力作!与法国秋季新版同步上市! 面对数字化时代的到来,美国、欧盟、中国、俄罗斯、印度、巴西、古巴、伊朗、南非、韩国、新加坡、肯尼亚、墨西哥、日本等世界各国各地区正在如何应对? 在国际地缘政治格局下如何把握数字化主流的大趋势? 谷歌、苹果、脸书、亚马逊、阿里巴巴、腾讯、中兴、华为等大家熟知的网络巨头接受了作者的采访。作者的......一起来看看 《智能》 这本书的介绍吧!

HTML 压缩/解压工具
HTML 压缩/解压工具

在线压缩/解压 HTML 代码

图片转BASE64编码
图片转BASE64编码

在线图片转Base64编码工具

html转js在线工具
html转js在线工具

html转js在线工具