使用XmlSerializer序列化可空属性
使用XmlSerializer可以方便的将对象序列化为xml,实现应用之间的数据交互。但是XmlSerializer却不能很好地序列化类型中的可空字段。
例如,有如下定义的类Person:
- [Serializable]
- [XmlRoot(ElementName = "Person")]
- public class Person
- {
- public string FirstName { get; set; }
- public string LastName { get; set; }
- public int? Age { get; set; }
- }
其中的Age属性为Nullable int类型。
我们的实例化代码如下所示:
- var person = new Person
- {
- FirstName = "First",
- };
- person.OutputXml(Console.Out);
其中方法OutputXml为扩展方法,使用XmlSerializer来序列化对象,具体定义为:
- public static void OutputXml<T>(this T dto, TextWriter textWriter)
- {
- var xmlTypeMapping = typeof(T);
- var serializer = new XmlSerializer(xmlTypeMapping);
- var xmlns = new XmlSerializerNamespaces();
- xmlns.Add(string.Empty, string.Empty);
- using (var writer = new XmlTextWriter(textWriter) { Formatting = Formatting.Indented })
- {
- serializer.Serialize(writer, dto, xmlns);
- }
- }
使用上述方法序列化对象person,得到的结果为:
注意到虽然Age属性为空,却仍然被序列化成 了古怪的xml,这往往不是所期望的结果。事实上同为空的LastName属性的序列化正是大多数情况下我们所期望的行为——忽略为空的属性。
另一方面,如果试图序列化类属性为xml 属性(而非xml元素),则甚至不能工作。例如如果我们将Person类的定义修改成如下形式以便序列化为xml属性:
- [Serializable]
- [XmlRoot(ElementName = "Person")]
- public class Person
- {
- [XmlAttribute]
- public string FirstName { get; set; }
- [XmlAttribute]
- public string LastName { get; set; }
- [XmlAttribute]
- public int? Age { get; set; }
- }
Xmlserializer甚至无法正常序列化上面同样的person对象并抛出如下“XmlAttribute/XmlText cannot be used to encode complex types”的错误:
如何解决上述的2个问题呢?
1. 序列化可空属性为XmlElement
为了在序列化可空属性的时候忽略空值的古怪输出,我们可以在Person类中定义一个返回bool的特殊方法ShouldSerializeAge,并在其中实现定义我们的序列化规则:
- [Serializable]
- [XmlRoot(ElementName = "Person")]
- public class Person
- {
- public string FirstName { get; set; }
- public string LastName { get; set; }
- public int? Age { get; set; }
- public bool ShouldSerializeAge()
- {
- return Age.HasValue;
- }
- }
在这里我们定义序列化规则为:序列化非空Age。注意该方法的名字一定是以ShouldSerialize开头并连接上需要自定义规则的属性名称。这样序列化出来的person对象为:
空的Age和空的LastName一样在序列化时被忽略了输出。正是我们期望的结果!
2. 序列化可空属性为XmlAttribute
由于可空属性无法被直接序列化为XmlAttribute,我们需要采用间接的办法——定义间接属性。此时我们可以如下定义Person类:
- [Serializable]
- [XmlRoot(ElementName = "Person")]
- public class Person
- {
- [XmlAttribute]
- public string FirstName { get; set; }
- [XmlAttribute]
- public string LastName { get; set; }
- [XmlIgnore] // (4)
- public int? Age { get; set; }
- [XmlAttribute(AttributeName = "Age")] // (1)
- public string AgeValue
- {
- get
- {
- // (2)
- return Age.HasValue ? Age.Value.ToString() : null;
- }
- set
- {
- int result;
- // (3)
- Age = int.TryParse(value, out result) ? result : (int?) null;
- }
- }
- }
注意类中注释的部分:
- 为原可空属性定义一个“虚拟”的属性,该属性可为任意名称任意类型(示例里定义的为string类型的AgeValue),但注意要将该属性的序列化名称置为原可空属性Age。
- 在get方法里根据Age是否为空将其转化为AgeValue。
- 在set方法里将输入的value转化为合适的值赋给可空属性Age。
- 将原Age属性标记为XmlIgnore,使XmlSerializer在序列化时忽略序列化该属性。
使用上述方法,我们可以将Age序列化为XmlAttribute了,虽然实际上我们是“骗”了XmlSerializer并使用另一个属性作为Age属性的值。
因此,当person对象中的Age为空时,我们得到如下的xml结果:
而当person对象中的Age不为空时,我们也可以正常的用XmlAttribute来表示Age的值了。 
使用XmlSerializer序列化可空属性的更多相关文章
- XmlSerializer序列化
XmlSerializer在命名空间using System.Xml.Serialization下. 序列化和反序列化的代码: using System.IO; using System.Xml; u ...
- C#.NET如何不序列化字段、属性
当我们使用公开属性以及公开字段时,都可以顺利的被序列化, 01.[Serializable] 02.public class MyClass 03.{ 04. public int ID; 05 ...
- .NET调用外部接口将得到的List数据,并使用XmlSerializer序列化List对象成XML格式
BidOpeningData.BidSupervisionSoapClient client = new BidOpeningData.BidSupervisionSoapClient(); Dict ...
- 第一章 JacksonUtil 序列化与反序列化属性总结
1.json-lib与Jackson 关于json-lib与Jackson对比总结如下: 1).性能方面,Jackson的处理能力高出Json-lib10倍左右. 2).json-lib已经停止更新, ...
- c# XML序列化与反序列化 属性字段标识
序列化对象 public class People { [XmlAttribute("NAME")] public string Name { set; get; } [XmlAt ...
- 学习C# XmlSerializer 序列化反序列化XML
类.变量常用头: [XmlRootAttribute]:对根节点的描述,在类声明中使用 如:下例的Html类 [XmlType]:对节点描述,在类声明中使用 如:下例的Head类 [X ...
- C# XmlSerializer序列化浅析
C# 中使用 XmlSerializer 实现类和xml文件的序列化和反序列化,使用起来非常简单. C# XmlSerializer实现序列化: XmlSerializer xml = new Xml ...
- C#调用XmlSerializer序列化时生成CDATA节点解决方法
public class Person{ public string Name { get; set; } public int Age { get; set; } } 引用 ...
- js如何判断Object是否为空?(属性是否为空)
js 判断一个 object 对象是否为空 转载原文 判断一个对象是否为空对象,本文给出三种判断方法: 1.最常见的思路,for...in... 遍历属性,为真则为“非空数组”:否则为“空数组” fo ...
随机推荐
- Java 基础【07】线程同步锁的选择
在需要线程同步的时候如何选择合适的线程锁? 例:选择可以存入到常量池当中的对象,String对象等 public class SyncTest { private String name = &quo ...
- HTML5+JS 《五子飞》游戏实现(八)人机对战
要想实现人机对战,就必须让电脑自动下棋,而且要知道自动去查找对方的棋子,看看有没有可以挑一对的,有没有可以夹一个的,这样下起来才有意思. 当电脑用户下完棋后,电脑应立即搜索用户的棋子,然后如果没有被吃 ...
- 备忘:maven 错误信息: Plugin execution not covered by lifecycle configuration
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/20 ...
- Redis的五种数据结构
Redis支持持久化只是它的一件武器,它提供了多达5种数据存储方式: 一 string(字符串) string是最简单的类型,你可以理解成与Memcached一模一样的类型,一个key对应一个val ...
- snr ber Eb/N0之间的区别与联系
信噪比(S/N)是指传输信号的平均功率与加性噪声的平均功率之比,载噪比(C/N)指已经调制的信号的平均功率与加性噪声的平均功率之比,它们都以对数的方式来计算,单位为dB. 对同一个传输系统而言,载噪比 ...
- jQuery问题集锦
[1]阻止提交表单 方法1: $(function () { $("input[type=submit]").click(function (event) { //如果不满足表单提 ...
- 5-touch 命令总结
- Vue-loader 开启压缩后的一些坑
在使用vue-loader 配合webpack 对.vue文件进行加载的时候,如果开启了代码压缩会出来下面几种问题,做个记录. 丢失td结束标记,导致页面的布局错乱 input的属性type为text ...
- springMvc对json的支持
实体类: public class User { private String id; //有这个注解的属性,不会转换为json @JsonIgnore private String name; .. ...
- Java创始人
詹姆斯·高斯林(英语:James Gosling,1955年5月19日-),出生于加拿大,软件专家,Java编程语言的共同创始人之一,一般公认他为“Java之父”. 在他12岁的时候,他已能设计电子游 ...