序列化特定类型时如何使 JSON.Net 序列化程序调用 ToString()?

Posted

技术标签:

【中文标题】序列化特定类型时如何使 JSON.Net 序列化程序调用 ToString()?【英文标题】:How to make JSON.Net serializer to call ToString() when serializing a particular type? 【发布时间】:2014-04-16 18:50:30 【问题描述】:

我正在使用 Newtonsoft.Json 序列化程序将 C# 类转换为 JSON。对于某些类,我不需要序列化器到单个属性的实例,而是只在对象上调用 ToString,即

public class Person

   public string FirstName  get; set; 
   public string LastName  get; set; 

   public override string ToString()  return string.Format("0 1", FirstName, LastName ); 

我应该怎么做才能将 Person 对象序列化为 ToString() 方法的结果?我可能有很多这样的类,所以我不想最终得到一个特定于 Person 类的序列化程序,我想要一个可以适用于任何类的序列化程序(我猜是通过属性)。

【问题讨论】:

【参考方案1】:

您可以使用自定义 JsonConverter 轻松做到这一点:

public class ToStringJsonConverter : JsonConverter

    public override bool CanConvert(Type objectType)
    
        return true;
    

    public override void WriteJson(JsonWriter writer, object value, JsonSerializer serializer)
    
        writer.WriteValue(value.ToString());
    

    public override bool CanRead
    
        get  return false; 
    

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

要使用转换器,请使用[JsonConverter] 属性装饰任何需要序列化为字符串的类,如下所示:

[JsonConverter(typeof(ToStringJsonConverter))]
public class Person

    ...

这是一个展示转换器的演示:

class Program

    static void Main(string[] args)
    
        Company company = new Company
        
            CompanyName = "Initrode",
            Boss = new Person  FirstName = "Head", LastName = "Honcho" ,
            Employees = new List<Person>
            
                new Person  FirstName = "Joe", LastName = "Schmoe" ,
                new Person  FirstName = "John", LastName = "Doe" 
            
        ;

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


public class Company

    public string CompanyName  get; set; 
    public Person Boss  get; set; 
    public List<Person> Employees  get; set; 


[JsonConverter(typeof(ToStringJsonConverter))]
public class Person

    public string FirstName  get; set; 
    public string LastName  get; set; 

    public override string ToString() 
     
        return string.Format("0 1", FirstName, LastName); 
    

输出:


  "CompanyName": "Initrode",
  "Boss": "Head Honcho",
  "Employees": [
    "Joe Schmoe",
    "John Doe"
  ]


如果您还需要能够从字符串转换回对象,您可以在转换器上实现ReadJson 方法,以便它查找public static Parse(string) 方法并调用它。注意:一定要把转换器的CanRead方法改成返回true(或者干脆删除CanRead重载),否则ReadJson永远不会被调用。

public override object ReadJson(JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer)

    MethodInfo parse = objectType.GetMethod("Parse", new Type[]  typeof(string) );
    if (parse != null && parse.IsStatic && parse.ReturnType == objectType)
    
        return parse.Invoke(null, new object[]  (string)reader.Value );
    

    throw new JsonException(string.Format(
        "The 0 type does not have a public static Parse(string) method that returns a 0.", 
        objectType.Name));

当然,要使上述方法起作用,您还需要确保在要转换的每个类上实现合适的Parse 方法(如果它尚不存在)。对于上面显示的 Person 类示例,该方法可能如下所示:

public static Person Parse(string s)

    if (string.IsNullOrWhiteSpace(s))
        throw new ArgumentException("s cannot be null or empty", "s");

    string[] parts = s.Split(new char[]  ' ' , 2);
    Person p = new Person  FirstName = parts[0] ;
    if (parts.Length > 1)
        p.LastName = parts[1];

    return p;

往返演示:https://dotnetfiddle.net/fd4EG4

【讨论】:

非常感谢。正是我需要的。像魅力一样工作。 @BrianRogers 感谢您的精彩回答。我想知道您是否可以就如何在转换器中实现 ReadJson() 提供任何见解,以便将字符串反序列化回其两个各自的属性? 一开始我发现这对我不起作用,但这是因为我的 ToString() 方法用“new”而不是“override”装饰(不知道为什么)——一旦我更正了它工作得很好。感谢分享。【参考方案2】:

如果不打算大规模使用,有一种更快的方法可以做到这一点,在下面的示例中,它是针对 RecordType 属性完成的

[JsonIgnore]
public RecordType RecType  get; set; 

[JsonProperty(PropertyName = "RecordType")]
private string RecordTypeString => RecType.ToString();

【讨论】:

【参考方案3】:

您可以简单地尝试 Newtonsoft 的 JSON 构建器库并使用这样的代码序列化 Person 类型的对象:

Dictionary<string, object> collection = new Dictionary<string, object>()
    
      "First", new Person(<add FirstName as constructor>),
      "Second", new Person(<add LastName as constructor>),

    ;
string json = JsonConvert.SerializeObject(collection, Formatting.Indented, new JsonSerializerSettings
  
    TypeNameHandling = TypeNameHandling.All,
    TypeNameAssemblyFormat = FormatterAssemblyStyle.Simple
  );

【讨论】:

谢谢。我采用了 Brian 建议的方法(使用自定义转换器)。【参考方案4】:

我没有时间测试我的解决方案,但它应该可以工作。假设您使用的所有类都是您自己的,为什么不对所有类进行 ToString 覆盖,而需要使用 Newtonsoft.Json 序列化程序的类可以在 ToString 方法中序列化并返回。这样,当你想要获取对象的序列化字符串时,你总是可以只调用 ToString 。

【讨论】:

首先,我不想混用 ToString 和 Serialize。他们有不同的目的。其次,我可以有复杂的对象图,所以我应该依靠序列化器来遍历图并应用序列化算法。用你的方法这是行不通的。

以上是关于序列化特定类型时如何使 JSON.Net 序列化程序调用 ToString()?的主要内容,如果未能解决你的问题,请参考以下文章

Json.NET:将嵌套数组反序列化为强类型对象

使用 Json.Net 进行 C# 枚举反序列化:将值转换为类型时出错

定制json序列化

Json.NET 是不是缓存类型的序列化信息?

C#JSON.NET - 反序列化使用不寻常数据结构的响应

如何使用 Json.NET 反序列化高精度十进制值?