.NET提供了很不错的XML序列化/反序列化器,(它们所在的命名空间为System.Xml.Serialization)这是很方便的,下面对它的使用做一些总结,以供参考。

1,简单序列化

public static string SerializeXml(object data) {
using (StringWriter sw = new StringWriter()) {
XmlSerializer xz = new XmlSerializer(data.GetType());
xz.Serialize(sw, data);
return sw.ToString();
}
}

以上代码是序列化为字符串,如果需要以流的形式返回序列化结果给客户端,或写入文件,那么通常需要选择一种编码,常见的编码格式是UTF-8,但某些特殊场合也许你会被要求使用GB2312编码,下面例子是使用GB2312编码的情况:

public static MemoryStream SerializeXml(object data) {
MemoryStream ms = new MemoryStream();
StreamWriter sw = new StreamWriter(ms, Encoding.GetEncoding("GB2312"));
XmlSerializer xz = new XmlSerializer(data.GetType());
xz.Serialize(sw, data);
return ms;
}

这样就直接把对象以特定编码格式序列化到MemoryStream里去了,当然也许你想到了,先使用前面的SerializeXml生成字符串,再把字符串以特定编码格式写到流或者字节数组中去不行吗?当然行,不过这样会多出一步,不够直接。

这里还有个要注意的地方,序列化到流的时候,不要对Stream及TextWriter对象包在using里,因为这样会导致流返回的时候已经被关闭。

2,简单反序列化

FileStream fs = File.Open("file.xml", FileMode.Open);
using (StreamReader sr = new StreamReader(fs, Encoding.UTF8)) {
XmlSerializer xz = new XmlSerializer(typeof(Department));
Department dept = (Department)xz.Deserialize(sr);
//blah blah ...
}

其中Department是你要反序列化出来的类,同样需要注意编码,这里指定的是UTF-8,但不排除有别的可能。

其实序列化和反序列化时可逆的,你通过怎样的类和编码把对象序列化成xml,就能通过怎样的类和编码将xml反序列化成对象。

3,指定XML标签的名字

[XmlRoot("department")]
public class Department {
public string DeptName { get; set; } [XmlElement("extra")]
public DeptExtraInfo DeptExtraInfo { get; set; }
}

通过XmlRoot注解和XmlElement注解即可实现,其中XmlRoot用于指定“根”,也就是XML的最上一层的Tag。

4,指定XML标签的属性

[XmlRoot("department")]
public class Department {
public string DeptName { get; set; } = "研发部"; [XmlAttribute("timestamp")]
public int Timestamp = ;
}

利用XmlAttribute注解,这么一来,Timestamp就成为了department这个根节点的timestamp属性。

5,去掉不需要的namespace

默认情况下,xml的头会带上类似这样的一个namespace:

<department xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:xsd="http://www.w3.org/2001/XMLSchema">
<!-- blah blah blah -->
</department>

你不需要的话可以修改一下序列化方法:

public static string SerializeXml(object data) {
using (Utf8Writer sw = new Utf8Writer()) {
XmlSerializer xz = new XmlSerializer(data.GetType());
XmlSerializerNamespaces ns = new XmlSerializerNamespaces();
ns.Add("", "");
xz.Serialize(sw, data, ns);
return sw.ToString();
}
}

6,序列化集合的时候不要“再包一层”

这个怎么说呢?先看这么一个类:

    [XmlRoot("department")]
public class Department {
public string DeptName { get; set; }; public List<Employee> Details { get; set; };
}

序列化出来的结果是:

<?xml version="1.0" encoding="utf-8"?>
<department>
<DeptName>研发部</DeptName>
<Employees>
<Employee>
<EmpName>张三</EmpName>
<EmpSalary>10000</EmpSalary>
</Employee>
<Employee>
<EmpName>李四</EmpName>
<EmpSalary>8000</EmpSalary>
</Employee>
</Employees>
</department>

注意Employee这个标签外面包了一层Employees,这个也许不是你想要的结果,这才是你想要的结果:

<?xml version="1.0" encoding="utf-8"?>
<department>
<DeptName>研发部</DeptName>
<Employee>
<EmpName>张三</EmpName>
<EmpSalary>10000</EmpSalary>
</Employee>
<Employee>
<EmpName>李四</EmpName>
<EmpSalary>8000</EmpSalary>
</Employee>
</department>

这个怎么做呢?很简单,在Employees前面加个XmlElement注解即可:

    [XmlRoot("department")]
public class Department {
public string DeptName { get; set; } = "研发部"; [XmlElement("Employee")]
public List<Employee> Employees { get; set; } = new List<Employee>();
}

另外,如果是只是想改一下之前的Employees标签的名字的话,用这样一个注解:[XmlArray("NewName")]。

7,序列化null值属性

默认情况下,null值的属性是不会被序列化的,想想看为什么?

因为生成<DeptName />这样的序列化结果的话,没办法知道DeptName到底是null还是空字符串,所以比较好的解决方法是在序列化之前,把null字符串填充为空字符串。可以考虑写一个帮助方法,利用反射遍历一个对象里的所有字符串属性,将null设置为空字符串,当然了,实际的情况要考虑得更全面点,比如对象里还有对象,而且还包含可枚举对象的情况,估计得使用递归。篇幅问题,代码我就不贴了。

另外还有一种比较地道的做法,不需要改变对象的值,那就是在对象上加上[XmlElement(IsNullable = true)]注解,但这样带来的问题就是会在序列化生成的tag中多出一个xsi:nil="true"这样的属性来。

8,手工反序列化

有些情况实在太特殊,没办法直接用简单的Deserialize方法来反序列化,例如这个XML:

<?xml version="1.0" encoding="UTF-8"?>
<ns0:DeliveryAddressUpdate_S10 xmlns:ns0="urn:ABC:GAIA:CN:LoadSetNoAndChineseDelAddr:ISC0186">
<Line>
<ASNNNB>95175154 </ASNNNB>
<CHDANR>00476</CHDANR>
<ASCUID>SHD3SHD3</ASCUID>
<IGAAUC>上海</IGAAUC>
<IGAAUC>闵行区</IGAAUC>
<IGAAUC>七莘路8888号</IGAAUC>
<IGAAUC>XXXX大楼XXXX室</IGAAUC>
</Line>
<Line>
<ASNNNB>124321 </ASNNNB>
<CHDANR>4321</CHDANR>
<ASCUID>4312</ASCUID>
<IGAAUC>上海</IGAAUC>
<IGAAUC>浦东新区</IGAAUC>
<IGAAUC>浦东大道9999号</IGAAUC>
<IGAAUC>YYYY大楼YYYY室</IGAAUC>
</Line>
</ns0:DeliveryAddressUpdate_S10>

首先根节点很奇葩,默认反序列化器不认,另外就是IGAAUC,重复多次,它的意图是说重复的这几个IGAAUC拼接在一起,生成一个地址,这个默认的反序列化显然做不到,手工读吧,参考代码如下:

List<Address> addrList = new List<Address>();
Address currentAddress = new Address();
XmlTextReader reader = new XmlTextReader(new MemoryStream(File.ReadAllBytes("test.xml")));
while (reader.Read()) {
if (reader.IsStartElement()) {
switch (reader.Name) {
case "Line":
currentAddress = new Address();
addrList.Add(currentAddress);
break;
case "ASNNNB":
currentAddress.Asnnb = reader.ReadString();
break;
case "CHDANR":
currentAddress.Chdanr = reader.ReadString();
break;
case "ASCUID":
currentAddress.Ascuid = reader.ReadString();
break;
case "IGAAUC":
currentAddress.Igaauc += reader.ReadString().Trim() + "\r\n";
break;
}
}
}
//addrList便是结果

C# XML序列化/反序列化参考的更多相关文章

  1. C# Json库 和 xml 序列化反序列化 存在的问题

    json 正常情况下不会对私有成员进行序列化和反序列化, 因此在用json做深拷贝时, 就会丢失数据. 解决办法:       声明成公有成员. json在序列化和反序列化时, 如果类中有IComma ...

  2. php json与xml序列化/反序列化

    在web开发中对象的序列化与反序列化经常使用,比较主流的有json格式与xml格式的序列化与反序列化,今天想写个jsop的小demo,结果发现不会使用php序列化,查了一下资料,做个笔记 简单数组js ...

  3. XML序列化反序列化—常用类

    public class XMLSerializer    {        #region (public) xml序列化        /// <summary>        /// ...

  4. C# 复杂格式多级深度XML序列化反序列化

     default.xml 文件如下: <config><partnerships>  <partnership name="Main_Listener" ...

  5. C# XML序列化/反序列化类XmlSerializer使用示例

    using System; using System.IO; using System.Text; using System.Xml; using System.Xml.Serialization; ...

  6. XML序列化反序列化

    using System; using System.Collections.Generic; using System.IO; using System.Xml.Serialization; nam ...

  7. XML序列化/反序列化数据库形式保存和读取。

    直接上码: 首先创建class1类 public class Class1 { public string name { get; set; } public int age { get; set; ...

  8. C# WPF xml序列化 反序列化

    using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.W ...

  9. XML序列化与反序列化接口对接实战,看这篇就够了

    关键字:c# .NET XML 序列化 反序列化 本文为接口对接实践经验分享,不对具体的XML概念定义进行阐述:涉及工具类及处理方法已在生产环境使用多年,可放心使用.当然如果你发现问题,或有不同想法, ...

随机推荐

  1. Linux kernel的中断子系统之(三):IRQ number和中断描述符

    返回目录:<ARM-Linux中断系统>. 总结: 二描述了中断处理示意图,以及关中断.开中断,和IRQ number重要概念. 三介绍了三个重要的结构体,irq_desc.irq_dat ...

  2. MIT KIT OpenID Connect Demo Client

    Hello world! You are NOT currently logged in. This example application is configured with several pa ...

  3. OAuth 2.0 认证的原理与实践

    摘要: 使用 OAuth 2.0 认证的的好处是显然易见的.你只需要用同一个账号密码,就能在各个网站进行访问,而免去了在每个网站都进行注册的繁琐过程. 本文将介绍 OAuth 2.0 的原理,并基于 ...

  4. Hibernate验证器

    第 4 章 Hibernate验证器  http://hibernate.org/validator/documentation/getting-started/#applying-constrain ...

  5. 关于数据库报Packet for query is too large (1986748 > 1048576)(mysql写入数据过大)的解决办法

    方法2 (很妥协,很纠结的办法) 进入mysql server 在mysql 命令行中运行 set global max_allowed_packet = 2*1024*1024*10 然后关闭掉这此 ...

  6. genymotion的安装及运行

    一.下载工具 安装genymontion一共需要下载三个东西,分别是genymotion.虚拟机virtualbox和ova 笔者提供百度云下载:mac版虚拟机 mac上genymotion.wind ...

  7. Nginx从听说到学会

    第一章 Nginx简介 Nginx是什么 没有听过Nginx?那么一定听过它的"同行"Apache吧!Nginx同Apache一样都是一种WEB服务器.基于REST架构风格,以统一 ...

  8. Netty与网络编程

    Netty什么? Netty项目是一个提供异步事件驱动网络应用框架和快速开发可维护的高性能高扩展性服务端和客户端协议工具集的成果.换句话说,Netty是一个NIO客户端服务端框架,它使得快速而简单的开 ...

  9. vue的常用组件方法应用

    项目技术: webpack + vue + element + axois (vue-resource) + less-loader+ ... vue的操作的方法案例: 1.数组数据还未获取到,做出预 ...

  10. 一个简单的例子实现自己的AOP

    AOP是Aspect Oriented Programming的缩写,意思是面向切面编程,与OOP(Object Oriented Programming)面向对象编程对等,都是一种编程思想. 从OO ...