C#三十一 序列化與反序列化

      網友投稿 824 2025-04-09

      序列化又稱串行化,是.NET運行時環境用來支持用戶定義類型的流化的機制。其目的是以某種存儲形成使自定義對象持久化,或者將這種對象從一個地方傳輸到另一個地方。


      .NET框架提供了兩種串行化的方式:1、是使用BinaryFormatter進行串行化;2、使用SoapFormatter進行串行化;3、使用XmlSerializer進行串行化。第一種方式提供了一個簡單的二進制數據流以及某些附加的類型信息,而第二種將數據流格式化為XML存儲;第三種其實和第二種差不多也是XML的格式存儲,只不過比第二種的XML格式要簡化很多(去掉了SOAP特有的額外信息)。

      可以使用[Serializable]屬性將類標志為可序列化的。如果某個類的元素不想被序列化,1、2可以使用[NonSerialized]屬性來標志,2、可以使用[XmlIgnore]來標志。

      視頻課堂https://edu.csdn.net/course/play/7621

      重點:

      ???????理解序列化 / 反序列化概念

      ???????能夠使用序列化/反序列化保持和恢復對象狀態

      預習功課:

      ???????序列化的概念

      ???????反序列化的概念

      ???????如何使用序列化和反序列化保持和恢復對象狀態

      ???????如何利用序列化來操作Xml文件

      8.1 序列化和反序列化簡介

      C#序列化和反序列化,兩者的程序處理方式基本一致,都是基于工廠模式的,所謂C#序列化就是是將對象轉換為容易傳輸的格式的過程,一般情況下轉化打流文件,放入內存或者IO文件中。例如,可以序列化一個對象,然后使用 HTTP 通過 Internet 在客戶端和服務器之間傳輸該對象,或者和其它應用程序共享使用。相反的,反序列化根據流重新構造對象。.NET自帶的有兩種序列化對象的方式,Xml和binary的,XML 序列化不轉換方法、索引器、私有字段或只讀屬性(只讀集合除外)。要序列化對象的所有字段和屬性(公共的和私有的),請使用 BinaryFormatter,而不要使用 XML 序列化。

      8.1.1 C#序列化和反序列化的實例應用剖析:

      二進制的C#序列化的方式:

      例如我們有個對象:

      1.????????????[Serializable]public?class?ClassToSerialize{

      2.????????????public?int?id=100;

      3.????????????public?string?name="Name";

      4.????????????}

      需要序列化該對象,必須在給該類加上Serializable的屬性,然后創建一個序列化寫入的流:FileStream fileStream = new FileStream("temp.dat",FileMode.Create);然后創建二進制格式器:BinaryFormatter b=newBinaryFormatter();然后是序列化:b.Serialize(fileStream,c);,然后關閉保存流。(可以見下面的例子)

      讀取一個已經被序列化的對象的時候:操作方式一樣,只是

      5.???????????FileStream?fileStream?=?new?FileStream(

      6.???????????"temp.dat",?FileMode.Open,

      7.???????????FileAccess.Read,?FileShare.Read);

      8.???????????ClassToSerialize?c?=

      9.???????????(ClassToSerialize)b.Deserialize(fileStream);

      然后就可以讀取了,完整的例子是:

      10.???????using?System;

      11.???????using?System.IO;

      12.???????using?System.Runtime.Serialization;

      13.???????using?System.Runtime.Serialization.Formatters.Binary;

      14.???????public?class?SerialTest{

      15.???????public?void?SerializeNow(){

      16.???????ClassToSerialize?c=new?ClassToSerialize();

      17.???????FileStream?fileStream?=?new?FileStream(

      18.???????"temp.dat",?FileMode.Create);

      19.

      20.???????BinaryFormatter?b=new?BinaryFormatter();

      21.???????b.Serialize(fileStream,c);

      22.???????fileStream.Close();

      23.???????}

      24.???????public?void?DeSerializeNow(){

      25.???????ClassToSerialize?c=new?ClassToSerialize();

      26.???????FileStream?fileStream?=?new?FileStream(

      27.???????"temp.dat",?FileMode.Open,

      28.????????FileAccess.Read,

      29.????????FileShare.Read);

      30.???????BinaryFormatter?b=new?BinaryFormatter();

      31.???????//SoapFormatter

      32.???????c=(ClassToSerialize)b.Deserialize(fileStream);

      33.???????Console.WriteLine(c.name);

      34.???????fileStream.Close();

      35.???????}

      36.???????public?static?void?Main(string[]?s){

      37.???????SerialTest?st=new?SerialTest();

      38.???????st.SerializeNow();

      39.???????st.DeSerializeNow();

      40.???????}

      41.???????}

      42.???????[Serializable]

      43.???????public?class?ClassToSerialize{

      44.???????public?int?id=100;

      45.???????public?string?name="Name";

      46.???????}

      這就是自帶的序列化和反序列的操作,但是,很多情況下,一個對象比較大,而且很多私有的屬性和方法我們不需要,例如在原型模式里面序列化的話,只需要序列Clone方法和一些屬性,私有的方法無需要,還例如在讀取大規模的IO的時候,讀取操作完全不需要... 這時候就需要自己集成重寫序列的ISerializable接口:

      實現該接口需要兩個注意的,一個就是構造函數,主要是為了反序列,另一個就是GetObjectData,主要是執行序列化,例如我們現在有一個Employee類需要序列化

      47.???????[Serializable()]

      48.???????//Set?this?attribute?to?all?the?classes?that?want?to?serialize

      49.???????public?class?Employee?:?ISerializable

      50.???????//derive?your?class?from?ISerializable?{

      51.???????public?int?EmpId;

      52.???????public?string?EmpName;

      53.???????[NonSerialized()]

      54.???????public?string?NoSerialString="NoSerialString-Test";

      55.

      56.???????}

      需要注意的是我這里的NoSerialString屬性前面有[NonSerialized()],就是說默認并不序列化這個屬性,而是使用默認值。

      首先是構造函數:

      57.???????public?Employee(SerializationInfo?info,?StreamingContext?ctxt)

      58.???????{

      59.???????EmpId?=?(int)info.GetValue(

      60.???????"EmployeeId",?typeof(int));

      61.???????EmpName?=?(String)info.GetValue(

      62.???????"EmployeeName",?typeof(string));

      63.???????//NoSerialString?=

      64.???????//(String)info.GetValue("NoSerialString",?typeof(string));

      65.???????}

      然后是C#序列化方法,就是當寫入流的時候怎么保存的:

      66.???????public?void?GetObjectData(SerializationInfo?info,?StreamingContext?ctxt)?{

      67.

      68.???????info.AddValue("EmployeeId",?EmpId);

      69.???????info.AddValue("EmployeeName",?EmpName);

      70.???????}

      把上面兩個方法寫入到Employee類,然后寫個測試的程序:

      71.???????public?class?ObjSerial{

      72.???????public?static?void?Main(String[]?args){

      73.???????Employee?mp?=?new?Employee();

      74.???????mp.EmpId?=?10;

      75.???????mp.EmpName?=?"Omkumar";

      76.???????mp.NoSerialString?=?"你好啊";

      77.

      78.??????????//C#序列化和反序列化之序列化

      79.???????Stream?stream?=?File.Open("EmployeeInfo.osl",?FileMode.Create);

      80.???????BinaryFormatter?bformatter?=?new?BinaryFormatter();

      81.

      82.???????Console.WriteLine("Writing?Employee?Information");

      83.???????bformatter.Serialize(stream,?mp);

      84.???????stream.Close();

      85.

      86.

      87.???????mp?=?null;

      88.??????????//C#序列化和反序列化之反序列

      89.???????stream?=?File.Open("EmployeeInfo.osl",?FileMode.Open);

      90.???????bformatter?=?new?BinaryFormatter();

      91.

      92.???????Console.WriteLine("Reading?Employee?Information");

      93.???????mp?=?(Employee)bformatter.Deserialize(stream);

      94.???????stream.Close();

      95.

      96.???????Console.WriteLine(

      97.???????"Employee?Id:?{0}",mp.EmpId.ToString());

      98.???????Console.WriteLine(

      99.???????"Employee?Name:?{0}",mp.EmpName);

      100.???? Console.WriteLine(

      101.???? "Employee?NoSerialString:?{0}",mp.NoSerialString);

      102.

      103.???? }

      C#三十一 序列化與反序列化

      104.???? }

      C#序列化和反序列化程序執行的結果是:

      105.???? Writing?Employee?Information

      106.???? Reading?Employee?Information

      107.???? Employee?Id:?10

      108.???? Employee?Name:?Omkumar

      109.???? Employee?NoSerialString:?NoSerialString-Test

      看到Employee NoSerialString:屬性的值沒有,它保持默認值,沒有序列化。

      8.2 Xml格式序列化及反序列化

      要序列化的對象的類:

      [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;

      }

      }

      進行序列化及反序列化的測試類:

      class Test

      {

      //序列化

      public void Serialiaze()

      {

      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 = newFileStream("c:\\cyj.xml", FileMode.Create, FileAccess.Write,FileShare.ReadWrite);

      xs.Serialize(stream, c);

      stream.Close();

      }

      //反序列化

      public void Deserialize()

      {

      XmlSerializer xs=new XmlSerializer(typeof(Person));

      Stream stream = newFileStream("c:\\cyj.xml", FileMode.Open, FileAccess.Read,FileShare.ReadWrite);

      Person p=(Person)xs.Deserialize(stream);

      Console.WriteLine(p.Name);

      Console.WriteLine(p.Age.ToString());

      Console.WriteLine(p.Courses.Length.ToString());

      Console.Read();

      }

      }

      格式化后Xml的文檔內容為:

      31

      英語

      數學

      cyj

      C# XML 彈性文件服務

      版權聲明:本文內容由網絡用戶投稿,版權歸原作者所有,本站不擁有其著作權,亦不承擔相應法律責任。如果您發現本站中有涉嫌抄襲或描述失實的內容,請聯系我們jiasou666@gmail.com 處理,核實后本網站將在24小時內刪除侵權內容。

      版權聲明:本文內容由網絡用戶投稿,版權歸原作者所有,本站不擁有其著作權,亦不承擔相應法律責任。如果您發現本站中有涉嫌抄襲或描述失實的內容,請聯系我們jiasou666@gmail.com 處理,核實后本網站將在24小時內刪除侵權內容。

      上一篇:【TensorFlow】01 TensorFlow簡介與Python基礎
      下一篇:有很多行數據,其中有空白行,如何快速刪除空白行?(如何批量刪除行與行之間的空白行)
      相關文章
      亚洲区日韩精品中文字幕| 国产大陆亚洲精品国产| 亚洲av无码成人精品区| 亚洲性无码AV中文字幕| 亚洲综合国产成人丁香五月激情 | 色欲aⅴ亚洲情无码AV| 亚洲AV无码男人的天堂| 亚洲av无码专区在线观看亚| 鲁死你资源站亚洲av| 极品色天使在线婷婷天堂亚洲 | 精品久久久久亚洲| 亚洲欧洲久久av| 亚洲成人高清在线| 在线日韩日本国产亚洲| 亚洲国产美女精品久久久久∴| 亚洲精品无码永久在线观看你懂的| 亚洲日韩中文无码久久| 久久亚洲精品中文字幕三区| 亚洲av永久无码精品表情包| 亚洲国产精品久久久久婷婷软件| 久久精品九九亚洲精品| 亚洲另类春色校园小说| 亚洲a级在线观看| 亚洲色欲色欲www在线播放 | 亚洲高清在线mv| 亚洲一区在线视频| 亚洲日韩精品国产3区| 亚洲a∨国产av综合av下载| 亚洲免费视频一区二区三区| 国产gv天堂亚洲国产gv刚刚碰| 亚洲第一极品精品无码久久| 亚洲精品在线免费看| 亚洲校园春色另类激情| 亚洲av日韩专区在线观看| 亚洲毛片不卡av在线播放一区| 国产亚洲一区二区精品| 亚洲精品视频在线播放| 亚洲熟妇无码八V在线播放| 国产成人高清亚洲一区久久| 国产亚洲精aa成人网站| 亚洲精选在线观看|