当前位置: 首页 > 面试题库 >

JSON.NET错误检测到类型的自引用循环

禄豪
2023-03-14
问题内容

我尝试序列化从实体数据模型.edmx自动生成的POCO类,使用时

JsonConvert.SerializeObject

我收到以下错误:

错误检测到类型为System.data.entity的自引用循环。

我该如何解决这个问题?


问题答案:

那是最好的解决方案 https://code.msdn.microsoft.com/Loop-Reference-handling-in-
caaffaf7

(我选择/尝试了这个,还有很多其他选择)

json.net序列化程序可以忽略循环引用。将以下代码放入WebApiConfig.cs文件中:

 config.Formatters.JsonFormatter.SerializerSettings.ReferenceLoopHandling 
= Newtonsoft.Json.ReferenceLoopHandling.Ignore;

简单的修复将使序列化程序忽略将导致循环的引用。但是,它具有局限性:

  • 数据丢失了循环参考信息
  • 该修复仅适用于JSON.net
  • 如果引用链很深,则无法控制引用级别

如果要在非API ASP.NET项目中使用此修复程序,可以将上述行添加到中Global.asax.cs,但首先添加:

var config = GlobalConfiguration.Configuration;

如果要在 .Net Core 项目中使用此功能,则可以更改Startup.cs为:

  var mvc = services.AddMvc(options =>
        {
           ...
        })
        .AddJsonOptions(x => x.SerializerSettings.ReferenceLoopHandling = Newtonsoft.Json.ReferenceLoopHandling.Ignore);

修复2:全局保留循环引用

此第二个修复程序类似于第一个。只需将代码更改为:

config.Formatters.JsonFormatter.SerializerSettings.ReferenceLoopHandling 
     = Newtonsoft.Json.ReferenceLoopHandling.Serialize;     
config.Formatters.JsonFormatter.SerializerSettings.PreserveReferencesHandling 
     = Newtonsoft.Json.PreserveReferencesHandling.Objects;

应用此设置后,数据形状将更改。

[
   {
      "$id":"1",
      "Category":{
         "$id":"2",
         "Products":[
            {
               "$id":"3",
               "Category":{
                  "$ref":"2"
               },
               "Id":2,
               "Name":"Yogurt"
            },
            {
               "$ref":"1"
            }
         ],
         "Id":1,
         "Name":"Diary"
      },
      "Id":1,
      "Name":"Whole Milk"
   },
   {
      "$ref":"3"
   }
]

$ id和$ ref保持所有引用并使对象图级别平坦,但是客户端代码需要知道形状变化以使用数据,并且它仅适用于JSON.NET序列化器。

修复3:忽略并保留引用属性

此修复是修饰模型类上的属性以控制模型或属性级别上的序列化行为。要忽略该属性:

 public class Category 
    { 
        public int Id { get; set; } 
        public string Name { get; set; }

        [JsonIgnore] 
        [IgnoreDataMember] 
        public virtual ICollection<Product> Products { get; set; } 
    }

JsonIgnore用于JSON.NET,而IgnoreDataMember用于XmlDCSerializer。要保留参考:

 // Fix 3 
        [JsonObject(IsReference = true)] 
        public class Category 
        { 
            public int Id { get; set; } 
            public string Name { get; set; }

           // Fix 3 
           //[JsonIgnore] 
           //[IgnoreDataMember] 
           public virtual ICollection<Product> Products { get; set; } 
       }

       [DataContract(IsReference = true)] 
       public class Product 
       { 
           [Key] 
           public int Id { get; set; }

           [DataMember] 
           public string Name { get; set; }

           [DataMember] 
           public virtual Category Category { get; set; } 
       }

JsonObject(IsReference = true)]适用于JSON.NET和[DataContract(IsReference = true)]XmlDCSerializer。注意:DataContract在类上应用后,您需要添加DataMember到要序列化的属性。

这些属性可以同时应用于json和xml序列化程序,并提供了对模型类的更多控制



 类似资料:
  • 问题内容: 上课: 和控制器代码: 它适用于LOCALHOST,但不适用于实时服务器: 错误: Json序列化类型的对象时检测到循环引用 我进行了搜索并找到了属性,因此将模型更改为 但是在实时服务器(win2008)上也会发生相同的错误。 如何避免该错误并成功序列化父数据? 问题答案: 尝试以下代码: …或者如果您仅需要父属性: 它并不是解决问题的真正方法,但在序列化DTO时是一种常见的解决方法。

  • 我有一个简单的类,如下所示: 但我收到以下错误消息: 检测到服务“App\Algorithm\Calculator”的循环引用,路径:“App\Algorithm\Calculator”- MatchService.php 问题是,但我到底做错了什么?

  • 我使用spring JAX-RS将文件作为多部分表单数据上传。我有inputstream对象作为参数。当我将这个输入流复制到磁盘时,我得到了我的文本文件。 但是当我试图检测这个输入流的内容类型时,我得到的内容类型是application/octet-stream。 作为一个测试,我还用tika desktop测试了相同的文件,并获得了正确的内容类型。

  • 问题内容: 我正在尝试做一个简单的JSON返回,但是我遇到以下问题。 我得到一个HTTP 500,但此问题的标题中显示了例外。我也试过 那也带来了同样的问题。 这是错误还是我的实现? 问题答案: 看来您的对象层次结构中有循环引用,而JSON序列化程序不支持。您是否需要所有列?您只能在视图中选择所需的属性: 这将使您的JSON对象更轻便,更易于理解。如果您有许多属性,则可以使用AutoMapper

  • 我正在尝试创建一个自定义注释。我有这样的注释: 我有一个方面的切入点: 但我得到了这个错误,我不明白为什么: org.springframework.security.config.annotation.configuration.objectPostProcessorConfiguration':bean初始化失败;嵌套异常为java.lang.IllegalArgumentException:

  • 问题内容: 所以这是我的交易 楷模 ....以下模型(这些模型已连接到EfDbContext)已连接到以下存储库… 接口/存储库 在我的HomeController()中,我得到了一个JsonResult方法,我想返回上下文。这是方法 杰森请求 我收到以下错误: 序列化类型为’System.Data.Entity.DynamicProxies.News_96C0B16EC4AC46070505EE