原文 www.cnblogs.com/mgen/archive/2011/12/03/2275014.html

注意:

运行代码请注意添加如下命名空间:

using System.Xml;

using System.Xml.Serialization;

using System.IO;

返回目录

1. 针对基类的XmlSerializer序列化派生类

派生类将会序列化成这样的XML:

<基类名称xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"

xmlns:xsd="http://www.w3.org/2001/XMLSchema"

xsi:type="派生类名称">

<!-- 基类和派生类的内容 -->

</基类名称>

第一种方法是在基类添加XmlInclude特性,这样的话基类的XmlSerializer可以序列化派生类了。

代码:

//基类加入XmlInclude

[XmlInclude(typeof(b))]

publicclassa

{

publicint aaa;

}

publicclassb : a

{

publicint bbb;

}

classProgram

{

staticvoid Main()

{

var xs =newXmlSerializer(typeof(a));

using (var textWriter =newStringWriter())

{

xs.Serialize(textWriter, newb());

Console.WriteLine(textWriter);

}

}

}

输出XML:

<axmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"

xmlns:xsd="http://www.w3.org/2001/XMLSchema"

xsi:type="b">

</aaa>

</bbb>

</a>

第二种方法就是在XmlSerializer的构造函数内指定派生类型,然后序列化,这样就不用在基类上加入XmlInclude特性了。

代码:

//无需加入XmlInclude

publicclassa

{

publicint aaa;

}

publicclassb : a

{

publicint bbb;

}

classProgram

{

staticvoid Main()

{

var xs =newXmlSerializer(typeof(a), newType[] { typeof(b) });

using (var textWriter =newStringWriter())

{

xs.Serialize(textWriter, newb());

Console.WriteLine(textWriter);

}

}

}

输出和上面一样!

返回目录

2. 类内成员是派生类的序列化

还是上面的类a和b,现在再加入一个c类,这个c类中有一个a的对象:

publicclassa

{

publicint aaa;

}

publicclassb : a

{

publicint bbb;

}

publicclassc

{

publica objA =newa();

}

此时直接XML序列化一个c对象,结果是这样的XML:

<cxmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"

xmlns:xsd="http://www.w3.org/2001/XMLSchema">

<objA>

</aaa>

</objA>

</c>

下面搞得复杂些,把c中的a对象改成这样:

publicclassc

{

publica objA =newb();

}

此时仿佛又回到了文章第一个标题,如果直接序列化c的话,会抛出异常。那么按照上面第一个方法,先在a上加入XmlInclude特性,然后再序列化c对象。

全部代码:

[XmlInclude(typeof(b))]

publicclassa

{

publicint aaa;

}

publicclassb : a

{

publicint bbb;

}

publicclassc

{

publica objA =newb();

}

classProgram

{

staticvoid Main()

{

var xs =newXmlSerializer(typeof(c));

using (var textWriter =newStringWriter())

{

xs.Serialize(textWriter, newc());

Console.WriteLine(textWriter);

}

}

}

输出XML(根节点c省略):

<objAxsi:type="b">

</aaa>

</bbb>

</objA>

和标题1的第一个方法类似!

但是标题1的第二个方法在这里没法使用的,因为c类和b类没有任何继承关系,在针对c的XmlSerializer不可能加入b的类型。这里其实还 有一种方法,加入XmlElement特性,其中ElementName属性是最终的Xml元素名称,而Type属性是针对的类型,这里加入b类型,不过 最好把a类型也加入(因为字段类型是a,有可能被赋值为a的对象)。

全部代码:

//不需要加XmlInclude

publicclassa

{

publicint aaa;

}

publicclassb : a

{

publicint bbb;

}

publicclassc

{

[XmlElement(ElementName ="b", Type =typeof(b))]

[XmlElement(ElementName ="a", Type =typeof(a))]

publica objA =newb();

}

classProgram

{

staticvoid Main()

{

var xs =newXmlSerializer(typeof(c));

using (var textWriter =newStringWriter())

{

xs.Serialize(textWriter, newc());

Console.WriteLine(textWriter);

}

}

}

生成XML:

<cxmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"

xmlns:xsd="http://www.w3.org/2001/XMLSchema">

<b>

</aaa>

</bbb>

</b>

</c>

 

作者:Mgen
出处:www.cnblogs.com/mgen

其他参考页面:我的软件和工程博客导读

.NET(C#):XML序列化时派生类的处理的更多相关文章

  1. c# 中xml序列化时相同节点存入不同类型值

    先上需要序列话的类定义: [System.Xml.Serialization.XmlIncludeAttribute(typeof(DescriptionType))] [System.CodeDom ...

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

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

  3. 在XML序列化时去除默认命名空间xmlns:xsd和xmlns:xsi

    摘 自: http://blog.csdn.net/fxhflower/article/details/7276820 可使用以下代码: //Create our own namespaces for ...

  4. C#中将xml文件反序列化为实例时采用基类还是派生类的问题

    基类: using System; using System.Collections.Generic; using System.Linq; using System.Text; namespace ...

  5. .NET中XML序列化和反序列化常用类和用来控制XML序列化的属性总结(XmlSerializer,XmlTypeAttribute,XmlElementAttribute,XmlAttributeAttribute,XmlArrayAttribute...)

    序列化和反序列化是指什么? 序列化(seriallization): 将对象转化为便于传输的数据格式, 常见的序列化格式:二进制格式,字节数组,json字符串,xml字符串.反序列化(deserial ...

  6. 关于C# XML序列化的一个BUG的修改

    关于C# XML序列化的一个BUG的修改 在我前一篇博客中提到用XML序列化作为数据库的一个方案,@拿笔小心 提到他们在用XML序列化时,遇到了一个比较严重的bug,即XML不闭合,系统不能正确的加载 ...

  7. Windows phone 之XML序列化与反序列化

    为什么要做序列化和反序列化? 一个回答: 我们都知道对象是不能在网络中直接传输的,不过还有补救的办法.XML(Extensible Markup Language)可扩展标记语言,本身就被设计用来存储 ...

  8. C#操作Xml:XmlSerializer 对象的Xml序列化和反序列化

    这篇随笔对应的.Net命名空间是System.Xml.Serialization:文中的示例代码需要引用这个命名空间. 为什么要做序列化和反序列化? .Net程序执行时,对象都驻留在内存中:内存中的对 ...

  9. Xml 序列化

    1 XML序列化只能序列化对象的公有属性,并且要求对象有一个无参的构造方法,否者无法反序列化. 2 [Serializable]和[NonSerialized]特性对XML序列化无效!所以使用XML序 ...

随机推荐

  1. android笔试题

    1.请谈一下Android系统的架构. 答:Android系统采用了分层架构,从高层到低层分别是应用程序层.应用程序框架层.系统运行库层和linux核心层.   2.谈谈android大众常用的五种布 ...

  2. 浅谈C中的指针和数组(六)

    数组和指针,原本不想在写了,觉得这部分差不多了,但是自己在写程序的时候还是发现了一个错误.首先说一下我的要求: 给一个函数传递一个二维数组,然后我想在这个函数里面计算这个数组的行数. 写个类似的错误D ...

  3. Linux网络管理——子网掩码

    1. 网络基础 .note-content {font-family: "Helvetica Neue",Arial,"Hiragino Sans GB",&q ...

  4. composer时间长了,提示需要升级,结果问题来了

    今天重新开始之前的laravel项目,结果composer提示需要升级,于是 composer selfupdate 结果提示无法连接目标库,查阅composer官网,发现之前的库地址已经启用了ssl ...

  5. Python中文显示问题

    默认pyhon使用ASCII码来解释程序的,默认不支持中文,需要在程序的第一行或者第二行声明编码. 官方解决方案:https://www.python.org/dev/peps/pep-0263/ T ...

  6. OSG报警特效学习总结

    方法一:粒子系统         OSG的粒子系统有自己定义好的模块,如osgParticle::ExplosionEffect(爆炸模拟):osgParticle::SmokeEffect(烟雾模拟 ...

  7. python encode和decode函数说明【转载】

    python encode和decode函数说明 字符串编码常用类型:utf-8,gb2312,cp936,gbk等. python中,我们使用decode()和encode()来进行解码和编码 在p ...

  8. SQL Server 日志截断

    截断事务日志是逻辑操作,只是把日志的一部分标记为‘不再需要’所以可以重用这个空间,截断不是物理操作,不会减少磁盘上文件的大小, 要减小物理大小必定要进行收缩. ----------- 有时就算是备份都 ...

  9. Tomcat 设置自动编译,自动发布,自动部署

    Tomcat服务器 具有一个常用的功能: 即自动编译,自动发布,自动部署功能. 问题: 当我们第一次发布程序以后,我们增删改Servelt,Java,.xml等文件,都必须重启Tomcat,如果项目巨 ...

  10. 对Msn协议的一点点研究

    这个也是好奇msn的协议还是2011年的时候写的, 就在网上找啊找啊, 可惜要不是不能用就是C++代码还有就是不完整, 到最后我也没弄成功,只到了下面这步就挂掉了...... 登录成功<=SBS ...