将TimeSpan值序列化和反序列化为Object类型的属性

时间:2014-12-10 19:45:50

标签: c# json serialization json.net

我有一个类,其类型为Object,可能包含TimeSpan对象。

我正在对此进行序列化并将其反序列化:

public class MyObject
{
    public object myTimeSpan { get; set; }
}

...

var myObject = new MyObject { myTimeSpan = TimeSpan.FromDays(2) };
var json = JsonConvert.SerializeObject(myObject);

...

var duplicateObject = JsonConvert.DeserializeObject<MyObject>(json);

当我这样做时,duplicateObject.myTimeSpan包含字符串&#34; 2:00:00&#34;。

如何将其反序列化为TimeSpan对象?

两点:

  1. 我没有处理已在其他地方序列化的字符串。我反序列化的唯一字符串是我序列化的字符串。
  2. 该字段必须是object类型 - 它可能包含字符串或DateTime,或者它可能包含TimeSpan。目前,它只是TimeSpan我遇到问题。

1 个答案:

答案 0 :(得分:5)

Json.Net具有TypeNameHandling设置,用于处理未知类型,以便可以正确反序列化。启用此设置后,它会导致Json.Net在JSON中插入特殊的$type属性,然后在反序列化时将其用作提示。不幸的是,这种设置似乎不适用于&#34;简单&#34;类型如TimeSpan,因为它们的值被序列化为字符串而不是对象。

要解决此问题,我建议制作一个使用相同想法的自定义JsonConverter。转换器将输出具有两个属性的子对象表示,而不是直接输出对象的字符串值:typevalue。子对象的type属性将包含对象的程序集限定类型名称,而value属性将包含实际的序列化值。在反序列化时,转换器将查找type属性以了解要从value属性实例化的对象类型。这种方法的好处是您不必向模型类添加任何其他属性或逻辑。

以下是代码的外观:

class UnknownObjectConverter : JsonConverter
{
    public override bool CanConvert(Type objectType)
    {
        return true;
    }

    public override void WriteJson(JsonWriter writer, object value, JsonSerializer serializer)
    {
        JObject jo = new JObject();
        jo["type"] = value.GetType().AssemblyQualifiedName;
        jo["value"] = JToken.FromObject(value, serializer);
        jo.WriteTo(writer);
    }

    public override object ReadJson(JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer)
    {
        JObject jo = JObject.Load(reader);
        Type type = Type.GetType(jo["type"].ToString(), throwOnError: true);
        return jo["value"].ToObject(type, serializer);
    }
}

要使用转换器,只需在类中装饰需要使用object属性进行特殊处理的所有[JsonConverter]属性,如下所示:

public class MyObject
{
    [JsonConverter(typeof(UnknownObjectConverter))]
    public object MyValue { get; set; }
}

这是一个往返演示,展示了它如何适用于几种不同的类型。

class Program
{
    static void Main(string[] args)
    {
        List<MyObject> list = new List<MyObject>
        {
            new MyObject { MyValue = TimeSpan.FromDays(2) },
            new MyObject { MyValue = "foo" },
            new MyObject { MyValue = new DateTime(2014, 12, 20, 17, 06, 44) },
            new MyObject { MyValue = new Tuple<int, bool>(23, true) }
        };

        string json = JsonConvert.SerializeObject(list, Formatting.Indented);
        Console.WriteLine(json);
        Console.WriteLine();

        list = JsonConvert.DeserializeObject<List<MyObject>>(json);
        foreach (MyObject obj in list)
        {
            Console.WriteLine(obj.MyValue.GetType().Name + ": " + obj.MyValue.ToString());
        }
    }
}

这是输出:

[
  {
    "MyValue": {
      "type": "System.TimeSpan, mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089",
      "value": "2.00:00:00"
    }
  },
  {
    "MyValue": {
      "type": "System.String, mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089",
      "value": "foo"
    }
  },
  {
    "MyValue": {
      "type": "System.DateTime, mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089",
      "value": "2014-12-20T17:06:44"
    }
  },
  {
    "MyValue": {
      "type": "System.Tuple`2[[System.Int32, mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089],[System.Boolean, mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089]], mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089",
      "value": {
        "Item1": 23,
        "Item2": true
      }
    }
  }
]

TimeSpan: 2.00:00:00
String: foo
DateTime: 12/20/2014 5:06:44 PM
Tuple`2: (23, True)
相关问题