c#序列化與反序列化概述
深入探討C#序列化和反序列化之前我們先要明白什麼是序列化,它又稱序列化,是.NET執行時環境用來支援使用者定義型別的流化的機制。序列化就是把一個物件儲存到一個檔案或資料庫欄位中去,反序列化就是在適當的時候把這個檔案再轉化成原來的物件使用。其目的是以某種儲存形成使自定義物件持久化,或者將這種物件從一個地方傳輸到另一個地方。.NET框架提供了兩種序列化的方式:1、是使用BinaryFormatter進行序列化;2、使用SoapFormatter進行序列化;3、使用XmlSerializer進行序列化。第一種方式提供了一個簡單的二進位制資料流以及某些附加的型別資訊,而第二種將資料流格式化為XML儲存;第三種其實和第二種差不多也是XML的格式儲存,只不過比第二種的XML格式要簡化很多(去掉了SOAP特有的額外資訊)。可以使用[Serializable]屬性將類標誌為可序列化的。如果某個類的元素不想被序列化,1、2可以使用[NonSerialized]屬性來標誌,2、可以使用[XmlIgnore]來標誌。
下面就讓我們開始深入瞭解C#序列化和反序列化:
C#序列化和反序列化1、使用BinaryFormatter進行序列化
下面是一個可序列化的類:
- using System;
- using System.Data;
- using System.Configuration;
- using System.Web;
- using System.Web.Security;
- using System.Web.UI;
- using System.Web.UI.WebControls;
- using System.Web.UI.WebControls.WebParts;
- using System.Web.UI.HtmlControls;
- using System.IO;
- using System.Runtime.Serialization.Formatters.Binary;
- /**//// ﹤summary﹥
- /// ClassToSerialize 的摘要說明
- /// ﹤/summary﹥
- [Serializable]
- public class ClassToSerialize
- {
- public int id = 100;
- public string name = "Name";
- [NonSerialized]
- public string Sex = "男";
- }
下面是序列化和反序列化的方法:
- public void SerializeNow()
- {
- ClassToSerialize c = new ClassToSerialize();
- FileStream fileStream =
- new FileStream("c:\\temp.dat", FileMode.Create);
- BinaryFormatter b = new BinaryFormatter();
- b.Serialize(fileStream, c);
- fileStream.Close();
- }
- public void DeSerializeNow()
- {
- ClassToSerialize c = new ClassToSerialize();
- c.Sex = "kkkk";
- FileStream fileStream =
- new FileStream("c:\\temp.dat",
- FileMode.Open, FileAccess.Read, FileShare.Read);
- BinaryFormatter b = new BinaryFormatter();
- c = b.Deserialize(fileStream) as ClassToSerialize;
- Response.Write(c.name);
- Response.Write(c.Sex);
- fileStream.Close();
- }
呼叫上述兩個方法就可以看到序列化的結果:Sex屬性因為被標誌為[NonSerialized],故其值總是為null。
C#序列化和反序列化2、使用SoapFormatter進行序列化
和BinaryFormatter類似,我們只需要做一下簡單修改即可:
a.將using語句中的.Formatter.Binary改為.Formatter.Soap;
b.將所有的BinaryFormatter替換為SoapFormatter.
c.確保報存檔案的副檔名為.xml
經過上面簡單改動,即可實現SoapFormatter的序列化,這時候產生的檔案就是一個xml格式的檔案。
C#序列化和反序列化3、使用XmlSerializer進行序列化
關於格式化器還有一個問題,假設我們需要XML,但是不想要SOAP特有的額外資訊,那麼我們應該怎麼辦呢?有兩中方案:要麼編寫一個實現IFormatter介面的類,採用的方式類似於SoapFormatter類,但是沒有你不需要的資訊;要麼使用庫類XmlSerializer,這個類不使用Serializable屬性,但是它提供了類似的功能。
如果我們不想使用主流的序列化機制,而想使用XmlSeralizer進行序列化我們需要做一下修改:
a.新增System.Xml.Serialization名稱空間。
b.Serializable和NoSerialized屬性將被忽略,而是使用XmlIgnore屬性,它的行為與NoSerialized類似。
c.XmlSeralizer要求類有個預設的構造器,這個條件可能已經滿足了。
下面看C#序列化和反序列化示例:
要序列化的類:
- using System;
- using System.Data;
- using System.Configuration;
- using System.Web;
- using System.Web.Security;
- using System.Web.UI;
- using System.Web.UI.WebControls;
- using System.Web.UI.WebControls.WebParts;
- using System.Web.UI.HtmlControls;
- using System.Xml.Serialization;
- [Serializable]
- public class Person
- {
- private string name;
- public string Name
- {
- get
- {
- return name;
- }
- set
- {
- name = value;
- }
- }
- public string Sex;
- public int Age = 31;
- public Course[] Courses;
- public Person()
- {
- }
- public Person(string Name)
- {
- name = Name;
- Sex = "男";
- }
- }
- [Serializable]
- public class Course
- {
- public string Name;
- [XmlIgnore]
- public string Description;
- public Course()
- {
- }
- public Course(string name, string description)
- {
- Name = name;
- Description = description;
- }
- }
C#序列化和反序列化方法:
- public void XMLSerialize()
- {
- Person c = new Person("cyj");
- c.Courses = new Course[2];
- c.Courses[0] = new Course("英語", "交流工具");
- c.Courses[1] = new Course("數學","自然科學");
- XmlSerializer xs = new XmlSerializer(typeof(Person));
- Stream stream = new FileStream("c:\\cyj.XML",FileMode.Create,FileAccess.Write,FileShare.Read);
- xs.Serialize(stream,c);
- stream.Close();
- }
- public void XMLDeserialize()
- {
- XmlSerializer xs = new XmlSerializer(typeof(Person));
- Stream stream = new FileStream("C:\\cyj.XML",FileMode.Open,FileAccess.Read,FileShare.Read);
- Person p = xs.Deserialize(stream) as Person;
- Response.Write(p.Name);
- Response.Write(p.Age.ToString());
- Response.Write(p.Courses[0].Name);
- Response.Write(p.Courses[0].Description);
- Response.Write(p.Courses[1].Name);
- Response.Write(p.Courses[1].Description);
- stream.Close();
- }
這裡Course類的Description屬性值將始終為null,生成的xml文件中也沒有該節點,如下:
- ﹤?xml version="1.0"?﹥
- ﹤Person xmlns:xsi=
- "http://www.w3.org/2001/XMLSchema-instance"
- xmlns:xsd="http://www.w3.org/2001/XMLSchema"﹥
- ﹤Sex﹥男﹤/Sex﹥
- ﹤Age﹥31﹤/Age﹥
- ﹤Courses﹥
- ﹤Course﹥
- ﹤Name﹥英語﹤/Name﹥
- ﹤Description﹥交流工具﹤/Description﹥
- ﹤/Course﹥
- ﹤Course﹥
- ﹤Name﹥數學﹤/Name﹥
- ﹤Description﹥自然科學﹤/Description﹥
- ﹤/Course﹥
- ﹤/Courses﹥
- ﹤Name﹥cyj﹤/Name﹥
- ﹤/Person﹥
C#序列化和反序列化4、自定義序列化
如果你希望讓使用者對類進行序列化,但是對資料流的組織方式不完全滿意,那麼可以通過在自定義類中實現介面來自定義序列化行為。這個介面只有一個方法,GetObjectData. 這個方法用於將對類物件進行序列化所需要的資料填進SerializationInfo物件。你使用的格式化器將構造SerializationInfo物件,然後在序列化時呼叫GetObjectData. 如果類的父類也實現了ISerializable,那麼應該呼叫GetObjectData的父類實現。如果你實現了ISerializable,那麼還必須提供一個具有特定原型的構造器,這個構造器的引數列表必須與GetObjectData相同。這個構造器應該被宣告為私有的或受保護的,以防止粗心的開發人員直接使用它。示例如下:
C#序列化和反序列化之實現ISerializable的類:
- using System;
- using System.Data;
- using System.Configuration;
- using System.Web;
- using System.Web.Security;
- using System.Web.UI;
- using System.Web.UI.WebControls;
- using System.Web.UI.WebControls.WebParts;
- using System.Web.UI.HtmlControls;
- using System.Runtime.Serialization;
- using System.Runtime.Serialization.Formatters.Binary;
- /**//// ﹤summary﹥
- /// Employee 的摘要說明
- /// ﹤/summary﹥
- [Serializable]
- public class Employee:ISerializable
- {
- public int EmpId=100;
- public string EmpName="劉德華";
- [NonSerialized]
- public string NoSerialString = "NoSerialString-Test";
- public Employee()
- {
- //
- // TODO: 在此處新增建構函式邏輯
- //
- }
- private Employee(SerializationInfo info, StreamingContext ctxt)
- {
- EmpId = (int)info.GetValue("EmployeeId", typeof(int));
- EmpName = (String)info.GetValue("EmployeeName",typeof(string));
- //NoSerialString = (String)info.GetValue("EmployeeString",typeof(string));
- }
- public void GetObjectData(SerializationInfo info, StreamingContext ctxt)
- {
- info.AddValue("EmployeeId", EmpId);
- info.AddValue("EmployeeName", EmpName);
- //info.AddValue("EmployeeString", NoSerialString);
- }
- }
C#序列化和反序列化方法:
- public void OtherEmployeeClassTest()
- {
- Employee mp = new Employee();
- mp.EmpId = 10;
- mp.EmpName = "邱楓";
- mp.NoSerialString = "你好呀";
- Stream steam = File.Open("c:\\temp3.dat", FileMode.Create);
- BinaryFormatter bf = new BinaryFormatter();
- Response.Write("Writing Employee Info:");
- bf.Serialize(steam,mp);
- steam.Close();
- mp = null;
- //C#序列化和反序列化之反序列化
- Stream steam2 = File.Open("c:\\temp3.dat", FileMode.Open);
- BinaryFormatter bf2 = new BinaryFormatter();
- Response.Write("Reading Employee Info:");
- Employee mp2 = (Employee)bf2.Deserialize(steam2);
- steam2.Close();
- Response.Write(mp2.EmpId);
- Response.Write(mp2.EmpName);
- Response.Write(mp2.NoSerialString);
- }
C#序列化和反序列化的深入探討就是一個體驗和嘗試的過程,那麼希望本文對你瞭解和學習C#序列化和反序列化有所幫助。
來自 “ ITPUB部落格 ” ,連結:http://blog.itpub.net/12639172/viewspace-630628/,如需轉載,請註明出處,否則將追究法律責任。
相關文章
- C#序列化和反序列化(json)C#JSON
- C# - XML讀寫與序列化C#XML
- 序列化與反序列化
- 序列化與反序列化(GO)Go
- C# Json反序列化C#JSON
- Java的序列化與反序列化Java
- 實體類與XML序列化與反序列化XML
- Java中的序列化與反序列化Java
- Flutter中JSON序列化與反序列化FlutterJSON
- Django REST framework 序列化與反序列化(4)DjangoRESTFramework
- Java物件的序列化與反序列化-Json篇Java物件JSON
- 物件序列化(序列化)物件
- Java安全基礎之Java序列化與反序列化Java
- jackson序列化與反序列化的應用實踐
- jackson對日期的處理(序列化與反序列化)
- Python學習——序列化與反序列化-json&picklePythonJSON
- Java序列化、反序列化、反序列化漏洞Java
- C#中實現JSON功能及物件的序列化和反序列化C#JSON物件
- C# xml文件反序列化記事C#XML
- 用遊戲來講序列化與反序列化機制遊戲
- 009 Rust 網路程式設計,序列化與反序列化Rust程式設計
- 關於序列化與反序列化的一些思考
- jackson對Exception型別物件的序列化與反序列化Exception型別物件
- Newtonsoft序列化與反序列化json字串使用方法總結JSON字串
- C# 中使物件序列化/反序列化 Json 支援使用派生型別以及泛型的方式C#物件JSON型別泛型
- 在C#中使用Json.Net進行序列化和反序列化及定製化C#JSON
- json序列化與反序列化 (map,struct, slice, 基本資料型別)JSONStruct資料型別
- Serializable詳解(1):程式碼驗證Java序列化與反序列化Java
- JSON繼承JsonConverter序列化與反序列化重寫類屬性JSON繼承
- C++ 序列化和反序列化C++
- Hive的序列化/反序列化(SerDe)Hive
- python的序列化和反序列化Python
- ctf serialize 序列化和反序列化
- Java的序列化和反序列化Java
- xml序列化和反序列化(一)XML
- 全網最適合入門的物件導向程式設計教程:57 Python字串與序列化-序列化與反序列化物件程式設計Python字串
- Protobuf-net:C#高效序列化工具,助力介面傳輸與前端解析C#前端
- Java IO: 序列化與ObjectInputStream、ObjectOutputStreamJavaObject
- Android序列化問題與思考Android