c# - Json.Net 从属性自定义序列化

标签 c# json serialization attributes json.net

我已经很好地阅读了 json.net 上的文档,但我已经没有想法了。我遇到过这样的情况,我的 child 引用了他们的 parent 。所以

public class ObjA
{
  public int Id {get;set}
  public string OtherStuff {get;set}
  public ObjB MyChild {get;set}
}

public class ObjB
{
  public int Id {get;set}
  public string OtherStuff {get;set}
  public ObjA MyParent {get;set}
}

这不会序列化。所以我可以做 [JsonIgnore] 但我宁愿做的是 [JsonIdOnly] 其中 ObjB,而不是对 MyParent 进行序列化或跳过 MyParent 完全显示 MyParentId:123 的 json 属性。所以

{OjbA:{
    Id:123,
    OtherStuff:some other stuff,
    MyChild:{
        Id:456,
        OtherStuff:Some other stuff,
        MyParentId:123,
        }
    }
 }

我知道我可以为某种类型编写一个自定义转换器。问题是我希望只有在指定时才会发生这种情况,否则我将无法序列化 ObjA。换句话说,只有当我用属性装饰它时,我才需要这种情况发生。所以

public class ObjB
{
  public int Id {get;set}
  public string OtherStuff {get;set}
  [JsonIdOnly]
  public ObjA MyParent {get;set}
}

最佳答案

如果您不关心 JSON 中包含一些额外的簿记信息,请将 JsonSerializerSettings 中的 PreserveReferenceHandling 选项设置为 All (或对象),正如@Athari建议的那样。这是使其发挥作用的最简单方法。如果您这样做,您的 JSON 将如下所示:

{
  "$id": "1",
  "Id": 123,
  "OtherStuff": "other stuff A",
  "MyChild": {
    "$id": "2",
    "Id": 456,
    "OtherStuff": "other stuff B",
    "MyParent": {
      "$ref": "1"
    }
  }
}

也就是说,有一种方法可以完成您最初想要的操作,即使用自定义 JsonConverter。您可以做的是创建一个转换器,它将接受任何具有 Id 属性的对象。然后,对于那些您希望仅将其序列化为 Id 的地方,您可以使用 [JsonConverter] 属性来修饰这些属性。然后,自定义转换器将用于这些情况,但不用于其他情况。转换器可能如下所示:

class IdOnlyConverter : JsonConverter
{
    public override bool CanConvert(Type objectType)
    {
        return objectType.IsClass;
    }

    public override void WriteJson(JsonWriter writer, object value, JsonSerializer serializer)
    {
        writer.WriteStartObject();
        writer.WritePropertyName("Id");
        writer.WriteValue(GetId(value));
        writer.WriteEndObject();
    }

    private int GetId(object obj)
    {
        PropertyInfo prop = obj.GetType().GetProperty("Id", typeof(int));
        if (prop != null && prop.CanRead)
        {
            return (int)prop.GetValue(obj, null);
        }
        return 0;
    }

    public override bool CanRead 
    { 
        get { return false; } 
    }

    public override object ReadJson(JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer)
    {
        throw new NotImplementedException();
    }
}

要使用转换器,您需要按照您概述的那样设置您的类。请注意 MyParent 如何使用属性进行修饰,以告诉 Json.Net 使用该属性的自定义转换器。

public class ObjA
{
    public int Id { get; set; }
    public string OtherStuff { get; set; }
    public ObjB MyChild { get; set; }
}

public class ObjB
{
    public int Id { get; set; }
    public string OtherStuff { get; set; }
    [JsonConverter(typeof(IdOnlyConverter))]
    public ObjA MyParent { get; set; }
}

序列化时,您需要将 JsonSerializerSettingsReferenceLoopHandling 选项设置为 Serialize,以告诉 Json.Net 在以下情况下不要抛出错误:检测到引用循环,并继续序列化(因为我们的转换器将处理它)。

将所有内容放在一起,下面是一些示例代码,演示了转换器的运行情况:

class Program
{
    static void Main(string[] args)
    {
        ObjA a = new ObjA();
        a.Id = 123;
        a.OtherStuff = "other stuff A";

        ObjB b = new ObjB();
        b.Id = 456;
        b.OtherStuff = "other stuff B";
        b.MyParent = a;

        a.MyChild = b;

        JsonSerializerSettings settings = new JsonSerializerSettings
        {
            ReferenceLoopHandling = Newtonsoft.Json.ReferenceLoopHandling.Serialize,
            Formatting = Newtonsoft.Json.Formatting.Indented
        };

        string json = JsonConvert.SerializeObject(a, settings);
        Console.WriteLine(json);
    }
}

这是上面的输出:

{
  "Id": 123,
  "OtherStuff": "other stuff A",
  "MyChild": {
    "Id": 456,
    "OtherStuff": "other stuff B",
    "MyParent": {
      "Id": 123
    }
  }
}

关于c# - Json.Net 从属性自定义序列化,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/19164012/

相关文章:

javascript - 构建一个 JSON 字符串

ruby-on-rails - JSON Rails API 的功能测试

java - 使用 Jackson 的 JSON 数据特殊字段

c# - 将域服务实现为存储库的扩展方法

c# - Request.Browser.IsMobileDevice 不适用于 iPadAir2 和 iOS 13.0.1

javascript - 我如何以这样的方式设计一个对象,我可以像这样制定一个 JSON?

java - 将A类序列化为B类java

c# - 在 .ts 中自动创建 .cs 类定义

c# - 从文件中高效读取结构化二进制数据

javascript - 奇怪的行为 - 序列化