在使用Entity Framework中使用WCF,在程序中调用服务一直报错,我一直以为是WCF的哪个地方的配置有问题,找来找去,一直没有解决。

        最后在网上找到一篇文章,说是序列化的问题,我才想起来,原来是跟Entity Framework有关。又是导航属性序列化的问题。觉得真是坑爹,早该想到的。

       分析出来,还是因为实体类中有导航属性,而EF会为有导航属性的实体类,生成一个代理类,而这个代理类也有自引用。所以WCF在序列化的时候,就会出错。报的就是上面的这个错误。

        一种办法说是禁用生成代理,这个是可以实现的,直接在Context里面禁用就可以了。

this.Configuration.ProxyCreationEnabled = false;

        但是如果禁用代理了,就不能通过导航属性得到关联的值了。所以这个方法不可行。在网上找到的另外一种办法就是自定义一个特性。个人开始嫌麻烦,不愿意自定义这个特性。

        我想是不是因为有这个导航属性才这样的,所以最开始想能不能让这个导航属性字段不序列化,只序列化其他的字段应该就行了,因为我前台并不需要用到这个导航属性。我在实体里面把导航属性前面的DataMember特性去掉了,最后发现还是报这个错误。后来设计发现,就算把这个特性去掉了,只是代表不序列化,但是这个属性还是有virtual关键字,EF框架还是会为它生成代理类。之所以WCF报错不是因为导航属性序列化的问题,而是因为有导航属性字段,会生成代理类。是代理类的序列化出了问题。

       最后还是参照网上的方法,自定义一个特性,然后在接口方法上应用这个特性,才把这个问题解决了。

       自定义特性的代码是:

        先定义一个类ProxyDataContractResolver:

//DataContractResolver在System.Runtime.Serialization这个命名空间里

public class ProxyDataContractResolver :DataContractResolver

{

    private XsdDataContractExporter _exporter = new XsdDataContractExporter();

 

    public override Type ResolveName(string typeName, string typeNamespace, Type declaredType,

                           DataContractResolver knownTypeResolver)

    {

        return knownTypeResolver.ResolveName(

                                   typeName, typeNamespace, declaredType, null);

    }

 

    public override bool TryResolveType(Type dataContractType, Type declaredType,

                           DataContractResolver knownTypeResolver,

                           out XmlDictionaryString typeName,

                           out XmlDictionaryString typeNamespace)

    {

 

        Type nonProxyType = ObjectContext.GetObjectType(dataContractType);

        if (nonProxyType != dataContractType)

        {

            // Type was a proxy type, so map the name to the non-proxy name

            XmlQualifiedName qualifiedName = _exporter.GetSchemaTypeName(nonProxyType);

            XmlDictionary dictionary = new XmlDictionary(2);

            typeName = new XmlDictionaryString(dictionary,

                                               qualifiedName.Name, 0);

            typeNamespace = new XmlDictionaryString(dictionary,

                                                     qualifiedName.Namespace, 1);

            return true;

        }

        else

        {

            // Type was not a proxy type, so do the default

            return knownTypeResolver.TryResolveType(

                                      dataContractType,

                                      declaredType,

                                      null,

                                      out typeName,

                                      out typeNamespace);

        }

    }

}

     再定义一个特性类ApplyProxyDataContractResolverAttribute

public class ApplyProxyDataContractResolverAttribute : Attribute, IOperationBehavior

{

    public void AddBindingParameters(OperationDescription description, BindingParameterCollection parameters)

    {

    }

 

    public void ApplyClientBehavior(OperationDescription description, ClientOperation proxy)

    {

        DataContractSerializerOperationBehavior

                   dataContractSerializerOperationBehavior =

                      description.Behaviors.Find<DataContractSerializerOperationBehavior>();

        dataContractSerializerOperationBehavior.DataContractResolver = new ProxyDataContractResolver();

    }

 

    public void ApplyDispatchBehavior(OperationDescription description, DispatchOperation dispatch)

    {

        DataContractSerializerOperationBehavior

                   dataContractSerializerOperationBehavior = description.Behaviors.Find<DataContractSerializerOperationBehavior>();

        dataContractSerializerOperationBehavior.DataContractResolver = new ProxyDataContractResolver();

    }

    public void Validate(OperationDescription description)

    {

    }

}

      然后在接口层的方法上面加上这个特性就可以了。

[OperationContract]

[ApplyProxyDataContractResolverAttribute]

       终于可以松一口气了,在使用EF的过程中,一定要考虑到这个导航属性的特殊性。

 

参考文章:

http://www.cnblogs.com/Gyoung/p/3153875.html(文中的代码都是摘自这篇文章,谢谢!)

Entity Framework + WCF 远程调用出错的更多相关文章

  1. Entity Framework + WCF REST JSON Service

    利用EF 和WCF 建立一个REST JSON Service. 首先我们要下载一个Visual Studio 的Template 叫 "ADO.NET C# POCO Entity Gen ...

  2. 使用Entity Framework时,序列化出错

             在使用Entity Framework时,如果数据库中有两个表是一对多或者是多对多的关系,那么生成的实体类中就有一个导航属性.这个导航属性前面都加上了一个virtual关键字.这个v ...

  3. feign远程调用出错

    如果你传递的参数,比较复杂时,默认会采用POST的请求方式. 传递单个参数时,推荐使用@PathVariable,如果传递的单个参数比较多,这里也可以采用@RequestParam,Feign接口中不 ...

  4. Entity Framework在WCF中序列化的问题

    问题描述 如果你在WCF中用Entity Framework来获取数据并返回实体对象,那么对下面的错误一定不陌生. 接收对 http://localhost:5115/ReService.svc 的 ...

  5. Entity Framework在WCF中序列化的问题(转)

    问题描述 如果你在WCF中用Entity Framework来获取数据并返回实体对象,那么对下面的错误一定不陌生. 接收对 http://localhost:5115/ReService.svc 的 ...

  6. Entity framework在用于WCF时创建数据模型的问题

    众所周知,WCF的传输对象,在创建时需要在类名上标识[DataContract]以及在属性上标识[DataMember],当我们在使用Entity framework时(不考虑Code first的情 ...

  7. 精进不休 .NET 4.5 (12) - ADO.NET Entity Framework 6.0 新特性, WCF Data Services 5.6 新特性

    [索引页][源码下载] 精进不休 .NET 4.5 (12) - ADO.NET Entity Framework 6.0 新特性, WCF Data Services 5.6 新特性 作者:weba ...

  8. ASP.NET MVC4 & Entity Framework 6.0 IIS 部署出错解决方案

    博客地址 http://blog.csdn.net/foxdave 近期了解MVC4的时候弄了一个简单的小工程,使用Entity Framework作为Model,F5启动调试运行的时候没有问题,但是 ...

  9. Metasploit远程调用Nessus出错

    Metasploit远程调用Nessus出错   从Nessus 7.1开始,Metaspliot远程调用Nessus创建新的扫描任务,会出现以下错误信息:   [*] New scan added ...

随机推荐

  1. HASH JION AND NESTED JION

    版权声明:本文为博主原创文章,未经博主同意不得转载. https://blog.csdn.net/sssbbbryj/article/details/27795905  关于HASH_JION的原 ...

  2. ZJOI2019Day1镇海中学游记(3.24~3.27)

    前言 第一次正式参加省选!不管怎么说,虽然明知自己很弱,但还是要尽力去尝试吧! 最好能进前\(100\),不然就没法去\(Day2\)了. \(Mar\ 24th\):出发 今天,我们正式从二中向宁波 ...

  3. 战神CPU计算机硬件组装

    今天本来更新DP常见优化的,但是下午土木学院吴书记找我组装电脑,晚上A题后,临时有一些事情,没来得及整理. 这里分享一下战神CPU(死垃圾)的组装. 一顿操作猛如虎,很艰难的装好机子了,发现吴书记被坑 ...

  4. WebApiConfig设置返回json并且对于get,post可以重名

    webapi2默认返回的是xml格式的,并且一个控制器中的方法名不能重名,列如:一个get,一个post这个也是不允许的,这些我们都可以进行设置. 下面设置:返回json格式,并且一个控制器中的方法可 ...

  5. SpeedTree制作超真实老宅

  6. video object detection

    先说一下,我觉得近两年最好的工作吧.其他的,我就不介绍了,因为我懂得少. 微软的jifeng dai的工作. Deep Feature Flow   github: https://github.co ...

  7. 用c#语言编写分解质因数

    using System;using System.Collections.Generic;using System.Linq;using System.Text;using System.Threa ...

  8. 关于Echarts的原生js获取DOM元素与动态加载DOM元素的冲突问题

    1.前言: 最近在做的看板项目,因为需要循环加载后台数据,并且用Echarts做数据呈现,所以jQuery和angular等库统统靠边站,Echarts用的是原生js获取DOM元素,至于诸多不兼容等深 ...

  9. Openresty最佳案例 | 第9篇:Openresty实现的网关权限控制

    转载请标明出处: http://blog.csdn.net/forezp/article/details/78616779 本文出自方志朋的博客 简介 采用openresty 开发出的api网关有很多 ...

  10. oracle聚簇表的理解 (转自:https://blog.csdn.net/gumengkai/article/details/51009345 )

    Oracle支持两种类型的聚簇:索引聚簇和哈希聚簇 一.索引聚簇表的原理 聚簇:如果一些表有一些共同的列,则将这样一组表存储在相同的数据块中 聚簇还表示把相关的数据存储在同一个块上.利用聚簇,一个块可 ...