Xml 简单序列化和返序列化

结构体都能进行 xml 的序列化和反序列化。

必须包含的命名空间:

1
2
3
4
using System.IO;
using System.Text;
using System.Xml;
using System.Xml.Serialization;

基本的序列化

这里以结构体类型为例 :

1
2
3
4
5
6
7
8
9
10
11
public struct Student
{
public string Name;
public int Age;

public Student(string name, int age)
{
this.Name = name;
this.Age = age;
}
}

这里访问类型必须要public

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
class XmlSerializerHelper
{
public void XmlSerial(object obj, string xmlFile)
{
// 序列化
using (MemoryStream ms = new MemoryStream())
{
XmlWriterSettings setting = new XmlWriterSettings() {
Encoding = new UTF8Encoding(false),
Indent = true,
};
using (XmlWriter writer = XmlWriter.Create(ms, setting))
{
XmlSerializer xmlSerializer = new XmlSerializer(obj.GetType());
// 去掉命名空间
XmlSerializerNamespaces ns = new XmlSerializerNamespaces();
ns.Add("", ""); // 命名空间赋值为空
xmlSerializer.Serialize(writer, obj, ns);

// 用StreamWriter
string xmlContent = Encoding.UTF8.GetString(ms.ToArray()); // 转成字符转
StreamWriter streamWriter = new StreamWriter(xmlFile);
streamWriter.Write(xmlContent);
streamWriter.Close();
}
}
}

// 返序列化
public object XmlDeserial(object obj, string xmlFile)
{
// 用 MemoryStream
string xmlString = File.ReadAllText(xmlFile); // xml 内容提取出来
using (MemoryStream MS = new MemoryStream(Encoding.UTF8.GetBytes(xmlString)))
{
using (XmlReader xr = XmlReader.Create(MS))
{
XmlSerializer xmlSerializer = new XmlSerializer(obj.GetType());
obj = xmlSerializer.Deserialize(xr); // 不要误写为 MS 流,有这个重载不会报错。但运行有错。
}
}

return obj;
}
}
1
2
3
4
5
6
7
8
9
10
11
12
class Program
{
static void Main(string[] args)
{
Student student = new Student() { Name = "毛泽东", Age = 18 };

XmlSerializerHelper xmlSerializerHelper = new XmlSerializerHelper();
xmlSerializerHelper.XmlSerial(student, "Student.xml");

Console.ReadKey();
}
}

输出,没什么问题:

1
2
3
4
5
<?xml version="1.0" encoding="utf-8"?>
<Student>
<Name>毛泽东</Name>
<Age>18</Age>
</Student>

简单反序列化

1
2
3
4
5
6
7
8
9
10
11
12
13
class Program
{
static void Main(string[] args)
{
XmlSerializerHelper xmlSerializerHelper = new XmlSerializerHelper();

Student stu;
stu = (Student)xmlSerializerHelper.XmlDeserial(new Student(), "Student.xml");
Console.WriteLine("Name = {0}\nAge = {1}", stu.Name, stu.Age);

Console.ReadKey();
}
}

输出,没什么问题:

1
2
Name = 毛泽东
Age = 18

测试一下嵌套序列化

1
2
3
4
5
6
7
8
9
10
11
12
public struct Student
{
public string Name;
public int Age;

public OtherInfo OtherInfo;
}
public struct OtherInfo
{
public int Birth;
public string Alisa;
}
1
2
3
4
5
6
7
8
9
10
11
12
13
class Program
{
static void Main(string[] args)
{
OtherInfo otherInfo = new OtherInfo() { Birth = 1893, Alisa = "毛润之"};
Student student = new Student() { Name = "毛泽东", Age = 18, OtherInfo = otherInfo };

XmlSerializerHelper xmlSerializerHelper = new XmlSerializerHelper();
xmlSerializerHelper.XmlSerial(student, "Student.xml");

Console.ReadKey();
}
}

输出:

1
2
3
4
5
6
7
8
9
<?xml version="1.0" encoding="utf-8"?>
<Student>
<Name>毛泽东</Name>
<Age>18</Age>
<OtherInfo>
<Birth>1893</Birth>
<Alisa>毛润之</Alisa>
</OtherInfo>
</Student>

测试下如果引用变量为null的情况

1
2
3
4
5
6
7
8
9
10
11
12
class Program
{
static void Main(string[] args)
{
Student student = new Student() ;

XmlSerializerHelper xmlSerializerHelper = new XmlSerializerHelper();
xmlSerializerHelper.XmlSerial(student, "Student.xml");

Console.ReadKey();
}
}

输出:

1
2
3
4
5
6
7
<?xml version="1.0" encoding="utf-8"?>
<Student>
<Age>0</Age>
<OtherInfo>
<Birth>0</Birth>
</OtherInfo>
</Student>

可见,null值不进行序列化,int类型会默认为0,所以存在。

struct改成class

1
2
3
4
5
6
7
8
9
10
11
12
public class Student
{
public string Name;
public int Age;

public OtherInfo OtherInfo;
}
public class OtherInfo
{
public int Birth;
public string Alisa;
}

输出:

1
2
3
4
<?xml version="1.0" encoding="utf-8"?>
<Student>
<Age>0</Age>
</Student>

class为引用类型,为 null 没输出了。

数组的序列化

输出:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
public class Person
{
public string Name { get; set; }
public int Age { get; set; }
public string Sex { get; set; }

[XmlElement("Dog")] // 不要“再包一层”
public List<Dog> Dog { get; set; }

public string Good { get; set; } = "100";
private string valuePrivate { get; set; } = "0";
}

public class Dog
{
public string Name { get; set; }
public int Age { get; set; }
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
class Program
{
static void Main(string[] args)
{
List<Dog> dogs = new List<Dog> {
new Dog() { Name = "小黑", Age = 9 },
new Dog() { Name = "小黄", Age = 10 }
};

Person person = new Person() {
Name = "孙悟空",
Age = 500,
Sex = "男",
Dog = dogs
};

XmlSerializerHelper xmlSerializerHelper = new XmlSerializerHelper();
xmlSerializerHelper.XmlSerial(person, "Person.xml");

Console.ReadKey();
}
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
<?xml version="1.0" encoding="utf-8"?>
<Person>
<Name>孙悟空</Name>
<Age>500</Age>
<Sex></Sex>
<Dog>
<Name>小黑</Name>
<Age>9</Age>
</Dog>
<Dog>
<Name>小黄</Name>
<Age>10</Age>
</Dog>
<Good>100</Good>
</Person>

可见:

  1. 数组如果不要再包一层,[XmlElement("Dog")] // 不要“再包一层”显示的命名个节点。不然输出如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
<?xml version="1.0" encoding="utf-8"?>
<Person>
<Name>孙悟空</Name>
<Age>500</Age>
<Sex></Sex>
<Dog>
<Dog>
<Name>小黑</Name>
<Age>9</Age>
</Dog>
<Dog>
<Name>小黄</Name>
<Age>10</Age>
</Dog>
</Dog>
<Good>100</Good>
</Person>

Dog 数组外面包了一层<Dog>

  1. private string valuePrivate { get; set; } = "0";私有成员没有被序列化。

感谢支持!