2

我正在使用 Json.Net 对具有 System.Text.Encoding 类型的属性的类进行序列化/反序列化。尝试一个简单的测试,我的课程没有任何问题被序列化:

public class TestClass {
    public Encoding TheEncoding { get; set; }
}

var testClass = new TestClass { TheEncoding = Encoding.UTF8 };
var json = JsonConvert.SerializeObject( testClass, Formatting.Indented );
var obj = JsonConvert.DeserializeObject<TestClass>( json );

序列化为:

{
  "TheEncoding": {
    "BodyName": "utf-8",
    "EncodingName": "Unicode (UTF-8)",
    "HeaderName": "utf-8",
    "WebName": "utf-8",
    "WindowsCodePage": 1200,
    "IsBrowserDisplay": true,
    "IsBrowserSave": true,
    "IsMailNewsDisplay": true,
    "IsMailNewsSave": true,
    "IsSingleByte": false,
    "EncoderFallback": {
      "DefaultString": "?",
      "MaxCharCount": 1
    },
    "DecoderFallback": {
      "DefaultString": "?",
      "MaxCharCount": 1
    },
    "IsReadOnly": true,
    "CodePage": 65001
  }
}

但是,在反序列化时,我遇到了一个异常:

无法创建 System.Text.Encoding 类型的实例。类型是接口或抽象类,不能实例化。路径“TheEncoding.BodyName”,第 3 行,位置 16。

通过创建一个处理 System.Text.Encoding 类型的自定义转换器,我能够解决这个问题:

public class JsonEncodingConverter : JsonConverter {
    public override void WriteJson( JsonWriter writer, object value, JsonSerializer serializer ) {
        // Serialize as the BodyName.
        serializer.Serialize( writer, ( value as Encoding ).BodyName );
    }

    public override object ReadJson( JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer ) {
        // Already good to go.
        return existingValue;
    }

    public override bool CanConvert( Type objectType ) {
        return ( typeof ( Encoding ).IsAssignableFrom( objectType ) );
    }
}

var testClass = new TestClass { TheEncoding = Encoding.UTF8 };
var json = JsonConvert.SerializeObject( testClass, Formatting.Indented, new JsonEncodingConverter() );
var obj = JsonConvert.DeserializeObject<TestClass>( json , new JsonEncodingConverter() );

现在使用自定义转换器进行序列化会产生:

{
  "TheEncoding": "utf-8"
}

当使用自定义转换器反序列化时,这个 JSON 可以成功地往返返回到原始对象。

我是 Json.Net 的新手,我有一种强烈的感觉,我正在努力做到这一点!肯定有更好、更少涉及的方式来处理 System.Text.Encoding 吗?

4

2 回答 2

2

使用[DataContract][DataMember]属性明确设置哪些属性将在您的TestClass对象上序列化。

  1. 省略[DataMember]属性中的TheEncoding属性。
  2. 创建一个被序列化为“utf-8”的辅助属性(取决于 的值TheEncoding)。包括此[DataMember]属性的属性。

例如:

[DataContract]
public class TestClass
{
  public Encoding TheEncoding { get; set; }

  [DataMember]
  public string TheEncodingName
  {
    get
    {
      if (this.TheEncoding == System.Text.Encoding.UTF8)
        return "utf-8";
      // TODO: More possibilities
    }
    set
    {
      if (value == "utf-8")
        this.TheEncoding = System.Text.Encoding.UTF8;
      // TODO: More possibilities
    }
  }
}

序列化时,TheEncoding将被跳过并TheEncodingName改为序列化。

于 2013-05-31T22:09:39.627 回答
0

我认为解决这个问题的更好方法是弄清楚你的 JSON 对象需要哪些属性?

服务器端没有理由包含Encoding对象的所有属性,因为作为 .NET 框架的一部分,您已经可以在运行时访问它。

在客户端上,我无法想象为什么您需要Encoding对象的所有属性。

EncodingName仅发送or和从中可以更直观地BodyName确定要在服务器上使用的编码对象。

例如,如果我要向服务器发送一些 JSON,指定utf-8我会知道使用Encoding.UTF8.

于 2013-05-31T22:04:13.393 回答