1. 概述

  应用程序间传递数据,需要先将数据对象转化为字符流或字节流的形式,然后接收端收到后再转化回原始的数据对象。这就是序列化与反序列化。

  本章介绍 .net中的序列化与反序列化、序列化器的种类 以及 为序列化配置对象。

2. 主要内容

  2.1 序列化与反序列化

    序列化只能保存对象的数据部分,不能保存方法部分。可以创建custom data transfer object(DTO)来只保存指定的数据信息。

    .net平台提供三种类型的序列化:

    ① XmlSerializer:

[Serializable]
public class Person
{
    public string FirstName { get; set; }
    public string LastName { get; set; }
    public int Age { get; set; }
}
XmlSerializer serializer = new XmlSerializer(typeof(Person));
string xml;
using (StringWriter stringWriter = new StringWriter())
{
    Person p = new Person
    {
        FirstName = “John”,
        LastName = “Doe”,
        Age = 
    };
    serializer.Serialize(stringWriter, p);
    xml = stringWriter.ToString();
} Console.WriteLine(xml); using (StringReader stringReader = new StringReader(xml))
{
    Person p = (Person)serializer.Deserialize(stringReader);
    Console.WriteLine(“{} {} is {} years old”, p.FirstName, p.LastName, p.Age);
}

      常用的几个attribute:   

      XmlIgnore
      XmlAttribute
      XmlElement
      XmlArray
        XmlArrayItem

[Serializable]
public class Person
{
    public string FirstName { get; set; }
    public string LastName { get; set; }
    public int Age { get; set; }
}
[Serializable]
public class Order
{
    [XmlAttribute]
    public int ID { get; set; }     [XmlIgnore]
    public bool IsDirty { get; set; }     [XmlArray(“Lines”)]
    [XmlArrayItem(“OrderLine”)]
    public List<OrderLine> OrderLines { get; set; }
} [Serializable]
public class VIPOrder : Order
{
    public string Description { get; set; }
} [Serializable]
public class OrderLine
{
    [XmlAttribute]
    public int ID { get; set; }     [XmlAttribute]
    public int Amount { get; set; }     [XmlElement(“OrderedProduct”)]
    public Product Product { get; set; }
} [Serializable]
public class Product
{
    [XmlAttribute]
    public int ID { get; set; }
    public decimal Price { get; set; }
    public string Description { get; set; }
}
private static Order CreateOrder()
{
    Product p1 = new Product { ID = , Description = “p2”, Price =  };
    Product p2 = new Product { ID = , Description = “p3”, Price =  };     Order order = new VIPOrder
    {
        ID = ,
        Description = “Order for John Doe. Use the nice giftwrap”,
        OrderLines = new List<OrderLine>
        { 
            new OrderLine { ID = , Amount = , Product = p1},
            new OrderLine { ID =  ,Amount = , Product = p2},
        }
    };     return order;
}
XmlSerializer serializer = new XmlSerializer(typeof(Order), 
new Type[] { typeof(VIPOrder) });
string xml;
using (StringWriter stringWriter = new StringWriter())
{
    Order order = CreateOrder();
    serializer.Serialize(stringWriter, order);
    xml = stringWriter.ToString();
}
using (StringReader stringReader = new StringReader(xml))
{
    Order o = (Order)serializer.Deserialize(stringReader);
    // Use the order
}

    ② binary serialization

[Serializable]
public class Person
{
    public int Id { get; set; }
    public string Name { get; set; }
    private bool isDirty = false;
} Person p = new Person
{
    Id = ,
    Name = “John Doe”
}; IFormatter formatter = new BinaryFormatter();
using (Stream stream = new FileStream(“data.bin”, FileMode.Create))
{
    formatter.Serialize(stream, p);
} using (Stream stream = new FileStream(“data.bin”, FileMode.Open))
{
    Person dp = (Person)formatter.Deserialize(stream);
}

      可以使用下列attributes来控制序列化与反序列化的过程

        OnDeserializedAttribute
        OnDeserializingAttribute
        OnSerializedAttribute
        OnSerializingAttribute

[Serializable]
public class Person
{
    public int Id { get; set; }
    public string Name { get; set; }     [NonSerialized]
    private bool isDirty = false;
    [OnSerializing()]
    internal void OnSerializingMethod(StreamingContext context)
    {
        Console.WriteLine(“OnSerializing.”);
    }   [OnSerialized()]
    internal void OnSerializedMethod(StreamingContext context)
    {
        Console.WriteLine(“OnSerialized.”);
    }     [OnDeserializing()]
    internal void OnDeserializingMethod(StreamingContext context)
    {
        Console.WriteLine(“OnDeserializing.”);
    }     [OnDeserialized()]
    internal void OnDeserializedMethod(StreamingContext context)
    {
        Console.WriteLine(“OnSerialized.”);
    }
}

        通过实现ISerializable接口,可以进行更精细的序列化控制。

[Serializable]
public class PersonComplex : ISerializable
{
    public int Id { get; set; }
    public string Name { get; set; }
    private bool isDirty = false;     public PersonComplex() { }
    protected PersonComplex(SerializationInfo info, StreamingContext context)
    {
        Id = info.GetInt32(“Value1”);
        Name = info.GetString(“Value2”);
        isDirty = info.GetBoolean(“Value3”);
    }     [System.Security.Permissions.SecurityPermission(SecurityAction.Demand, 
SerializationFormatter = true)]
    public void GetObjectData(SerializationInfo info, StreamingContext context)
    {
        info.AddValue(“Value1”, Id);
        info.AddValue(“Value2”, Name);
        info.AddValue(“Value3”, isDirty);
    }
}

    ③ 使用DataContract

      WCF中使用DataContract来将对象序列化为Xml或者Json格式。

[DataContract]
public class PersonDataContract
{
    [DataMember]
    public int Id { get; set; }
    [DataMember]
    public string Name { get; set; }     private bool isDirty = false;
}

    ④ 使用Json序列化器

[DataContract]
public class Person
{
    [DataMember]
    public int Id { get; set; }
    [DataMember]
    public string Name { get; set; }
} Person p = new Person
{
    Id = ,
    Name = “John Doe”
}; using (MemoryStream stream = new MemoryStream())
{
    DataContractJsonSerializer ser = new DataContractJsonSerializer(typeof(Person));
    ser.WriteObject(stream, p);     stream.Position = ;
    StreamReader streamReader = new StreamReader(stream);
    Console.WriteLine(streamReader.ReadToEnd()); // Displays {“Id”:1,”Name”:”John Doe”}    stream.Position = ;
    Person result = (Person)ser.ReadObject(stream);
}

3. 总结

  ① 序列化 是将一个对象转化为一个字符流或者字节流的过程。反序列化正好相反。

  ② 可以使用XmlSerializer来实现Xml格式的序列化。还可以使用指定的attributes来配置序列化操作。

  ③ 可以使用BinaryFormatter来实现二进制格式的序列化。

  ④ Wcf中使用DataContractSerializer来配置序列化操作。

  ⑤ 使用DataContractJsonSerializer来创建轻型文本格式Json。

第二十一章 数据访问(In .net4.5) 之 序列化的更多相关文章

  1. 第二十二章 数据访问(In .net4.5) 之 集合

    1. 概述 本章内容包括 .net平台中的集合.如何选择集合 以及 如何实现自定义集合. 2. 主要内容 2.1 使用数组(Array) ]; ; x < arrayOfInt.Length;  ...

  2. 第十九章 数据访问(In .net4.5) 之 处理数据

    1. 概述 本章介绍 数据库.Json和Xml.web services 三种介质上的数据操作. 2. 主要内容 2.1 数据库 ① 建立连接 .net平台中的数据连接类都继承自DbConnectio ...

  3. JavaScript高级程序设计:第二十一章

    第二十一章 Ajax与Comet 一.XMLHttpRequest对象 1.XHT的用法 在使用XHR对象时,要调用的第一个方法时open( ),它接受3个参数:要发送的请求的类型.请求的URL和表示 ...

  4. 2017.2.15 开涛shiro教程-第二十一章-授予身份与切换身份(二) controller

    原博客地址:http://jinnianshilongnian.iteye.com/blog/2018398 根据下载的pdf学习. 开涛shiro教程-第二十一章-授予身份与切换身份(二) 1.回顾 ...

  5. 2017.2.15 开涛shiro教程-第二十一章-授予身份与切换身份(一) table、entity、service、dao

    原博客地址:http://jinnianshilongnian.iteye.com/blog/2018398 根据下载的pdf学习. 第二十一章 授予身份与切换身份(一) 1.使用场景 某个领导因为某 ...

  6. Introduction to 3D Game Programming with DirectX 12 学习笔记之 --- 第二十一章:环境光遮蔽(AMBIENT OCCLUSION)

    原文:Introduction to 3D Game Programming with DirectX 12 学习笔记之 --- 第二十一章:环境光遮蔽(AMBIENT OCCLUSION) 学习目标 ...

  7. Flask 教程 第二十一章:用户通知

    本文翻译自The Flask Mega-Tutorial Part XXI: User Notifications 这是Flask Mega-Tutorial系列的第二十一章,我将添加一个私有消息功能 ...

  8. Gradle 1.12用户指南翻译——第二十一章. Gradle 插件

    昨天晚上只顾着和女朋友看<匆匆那年>电视剧的最后几集,所以说好的Android文档<Gradle 插件用户指南>第五章自然也没翻译多少.所以今天也发不了第五章的翻译了,就发几个 ...

  9. 第二十一章 Django的分页与cookie

    第二十一章 Django的分页与cookie 第一课 模板 1.模板的继承 在Template目录下新建模板master.html <!DOCTYPE html> <html lan ...

随机推荐

  1. java 实例之杨辉三角

    public class study{ public static void main(String args[]){ int i,j,level=7; int Yang[][] = new int[ ...

  2. Kafka Quick Start

    1.Download > tar -xzf kafka_2.11-0.10.0.0.tgz> cd kafka_2.11-0.10.0.0 2.启动zookeeper服务 Kafka使用的 ...

  3. extjs grid 分页

    在使用extjs创建带分页功能的 grid 如下: 1.创建一个模型 // 创建算定义模型 模型名称 User Ext.define('User', { extend: 'Ext.data.Model ...

  4. 网站导航不止有hao123!

    网址导航对于我们上网而言非常的重要,在一定程度上决定了我们每天都在接触一些什么样的网络信息.我个人一直用的是hao123,总体感觉这个网址导航是非常的不错的,不过网址导航不只只有这一个好的更不只有这一 ...

  5. Android控件大全(一)——DialogFragment创建对话框

    DialogFragment在android 3.0时被引入.是一种特殊的Fragment,用于在Activity的内容之上展示一个模态的对话框.典型的用于:展示警告框,输入框,确认框等等. 在Dia ...

  6. 【EF 4】ORM框架及其流行产品之一EF介绍

    导读:跳进了多租户切换数据库的坑,那么就继续走下去吧.在我们的项目中,是运用EF实现对数据库的操作,那么EF其实是.NET系统中,基于ORM框架的一个产品实现.在java那边,则有Hibernate和 ...

  7. CreateThread和_BeginThread的区别

    1.程序: 程序构成: (1)源代码 (2)可执行的二进制代码 程序是指令和数据的有序集合,其本身没有任何运行的含义,是一个静态的概念.由操作系统加载其可执行的二进制代码,分配相应的数据结构:进程控制 ...

  8. linux 内存使用

    # df -h Filesystem Size Used Avail Use% Mounted on /dev/sda1 50G 1.9G 45G 5% / tmpfs 1.9G 0 1.9G 0% ...

  9. WWF3事件类型活动<第三篇>

    WWF将工作流分为两大类: 面向Human:在工作流运行时通过用户对外部应用程序的操作来影响工作流的业务流转. 面向System:应用程序控制流程. 工作流与应用程序都是可以单独存在的,因此它们之间的 ...

  10. 1.4Linux内核版本号的定义规则

    Linux内核版本号的组成: (1)主版本号: (2)次版本号: (3)修订版本号: (4)微调版本号: (5)为特定的Linux系统特别调校的描述: 例子:2.6.29.7-flykernel-12 ...