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

 更新时间:2013年05月14日 11:01:45   作者:  
本篇文章对Silverlight调用WCF/Rest异常的解决方法进行了详细的分析介绍,需要的朋友参考下

新建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)
{
}


运行,结果如下:

image

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

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

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

复制代码 代码如下:

public App()
        {
            bool bRegisterPrefix = WebRequest.RegisterPrefix(http://localhost:9541/,

WebRequestCreator.ClientHttp);
            //other Code
        }


再次运行代码:image

这是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

image

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

image

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

注意:别忘记了添加跨域和授权文件: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

image

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

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

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

image

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

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

image

因为我们尝试反序列化为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);

重新运行:

image

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

最后没办法既然有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的过程是可以被封装的,读者自己尝试下吧,呵呵。

结果如下图:

image

相关文章

  • C#使用Twain协议实现扫描仪连续扫描功能

    C#使用Twain协议实现扫描仪连续扫描功能

    这篇文章主要介绍了C#使用Twain协议实现扫描仪连续扫描,只需一行代码,就可实现一次扫描多张,且不需要更改扫描仪的任何设置,需要的朋友可以参考下
    2022-01-01
  • C#实现异步操作的几种方式

    C#实现异步操作的几种方式

    在C#中,异步操作可以提高程序的性能和响应能力,本文主要介绍了C#实现异步操作的几种方式,具有一定的参考价值,感兴趣的可以了解一下
    2024-03-03
  • OpenCvSharp图像的修改和保存以及掩膜操作

    OpenCvSharp图像的修改和保存以及掩膜操作

    这篇文章主要介绍了OpenCvSharp图像的修改和保存以及掩膜操作,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-11-11
  • C# 设计模式系列教程-简单工厂模式

    C# 设计模式系列教程-简单工厂模式

    简单工厂模式职责单一,实现简单,且实现了客户端代码与具体实现的解耦。
    2016-06-06
  • C# RichTextBox制作文本编辑器

    C# RichTextBox制作文本编辑器

    这篇文章主要为大家详细介绍了C# RichTextBox制作文本编辑器的方法,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-04-04
  • C#窗体全屏功能实例代码

    C#窗体全屏功能实例代码

    简要介绍窗体全屏的方法,列出实现窗体的全屏功能以及全屏功能的快捷键的代码,需要的朋友可以参考下
    2016-11-11
  • 使用VS2019生成C#应用安装包的方法步骤

    使用VS2019生成C#应用安装包的方法步骤

    本文主要介绍了使用VS2019生成C#应用安装包的方法步骤,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2022-05-05
  • C#实现位图转换成图标的方法

    C#实现位图转换成图标的方法

    这篇文章主要介绍了C#实现位图转换成图标的方法,可实现将bmp格式位图转换成ico格式图标的功能,需要的朋友可以参考下
    2015-06-06
  • Winform使用DataGridView实现下拉筛选

    Winform使用DataGridView实现下拉筛选

    这篇文章主要为大家详细介绍了Winform如何使用原生DataGridView实现下拉筛选功能,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起了解一下
    2023-09-09
  • DirectInfo.GetFiles返回数组的默认排序示例

    DirectInfo.GetFiles返回数组的默认排序示例

    这篇文章主要介绍了,DirectInfo.GetFiles返回数组的默认排序示例NTFS和CDFS下,是按照字母顺序,而FAT下,按照文件创建时间顺序
    2014-01-01

最新评论