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. 10gR2-11gR1,11gR2如何干净的清除并重建OCR和表决磁盘

    下面分别讨论10gR2-11gR1和11gR2干净的清除并重建OCR和表决磁盘的方法. 一.10gR2-11gR1干净的清除并重建OCR和表决磁盘的方法 参考METALINK文章:ID 399482. ...

  2. 翻译:为 URL Rewrite 模块创建重写规则

    原文名称:Creating Rewrite Rules for the URL Rewrite Module 原文地址:http://www.iis.net/learn/extensions/url- ...

  3. JAVA中求解对象所占字节大小

    该类为cache4j缓存框架中的工具类方法,该方法实现了两个接口 接口1:计算对象在内存中所占字节数 接口2:复制对象,实现深度克隆效果,实现原理为先序列化对象,然后在反序列化对象:返回一个新的对象, ...

  4. linux启动init流程(转)

    当系统启动时,首先启动内核,内核调用init来完成引导进程.init启动时,它会在/etc/inittab内查找默认的运行级别:如id:2:initdefault:运行/etc/rc.d/init.d ...

  5. MATLAB mex文件

    MATLAB的mex文件是一种特征的函数封装形式,这类函数一般由C/C++语言编写的,经过MATLAB编译器处理而生成的二进制文件.它是可以被MATLAB解释器自动装载并执行的动态链接程序,类似于wi ...

  6. 8051学习笔记——AD

    AD.C #include<reg52.h> #include <iic.h> #define PCF8591 0x90 //PCF8591 地址 sbit LS138A=P2 ...

  7. extjs grid 分页

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

  8. js控制台调试

    在web编程的过程中js代码出现错误,可以通过console.log()将相关信息输入到控制台进行调试. 清空控制台右击选 Clear console 菜单,或者输入 clear() 都行 控制台相关 ...

  9. 学习练习 java产生6个不同的数字

    public static void main(String[] args) { Random r=new Random(); int arr[]=new int[6]; for(int i=0;i& ...

  10. 洛谷P1613 跑路

    P1613 跑路 176通过 539提交 题目提供者该用户不存在 标签倍增动态规划 难度普及+/提高 提交该题 讨论 题解 记录 最新讨论 这个题的数据.. 题意问题 表意 题目描述 小A的工作不仅繁 ...