成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

淺談C++應(yīng)用程序中創(chuàng)建持久化對象

開發(fā) 后端
本文介紹的是在C++應(yīng)用程序中,怎么創(chuàng)建持久化對象,希望對你有幫助,一起來看。

持久化的對象,是已經(jīng)存儲到數(shù)據(jù)庫或保存到本地硬盤中的對象,我們稱之為持久化對象。下面介紹C++中的持久化對象。

持久對象(persistent objects)廣泛應(yīng)用于游戲、分布式數(shù)據(jù)庫系統(tǒng)、多媒體以及圖形應(yīng)用程序中。目前C++并不直接支持持久性(persistence)(但有一些在C++未來版本中添加持久性和反射(reflection)的建議)。

持久對象可以在創(chuàng)建它的程序的作用域之外保持自身狀態(tài)。把對象寫入一個文件并在以后重建之,或者把對象傳送到一臺遠程機器,就是這樣的例子。對持久性的支持并不象***眼看上去那樣簡單,同一對象的大小和內(nèi)存布局在不同的平臺上可能并不相同,而不同的字節(jié)次序(byte ordering),或稱為endian-ness,使事情更加復(fù)雜化。

在下文中我將討論如何實現(xiàn)持久性,而無須求助于DCOM和 CORBA之類的第三方框架。對于小型和可移植的應(yīng)用程序而言,這是一種有效并令人滿意的方案。

序列化(serialization)基礎(chǔ)

為了使一個對象持久存在,必須把它的狀態(tài)保存在非易失的存儲設(shè)備中。考慮一個錄制和播放MP3文件的應(yīng)用程序,每首單曲都表示為一個包含標(biāo)題、唱片、歌手、時間、速率、錄制日期以及相應(yīng)的 MP3文件的對象,該應(yīng)用程序在跟蹤列表中顯示最近播放的曲目。你的目標(biāo)是通過序列化,也就是把對象寫入一個文件,使MP3對象成為持久對象,同時通過反序列化(deserialization)在下一個 session中重建這些對象。

序列化內(nèi)置數(shù)據(jù)類型

每個對象最終都由內(nèi)置數(shù)據(jù)成員組成,如int, bool, char[]等等。你的***個任務(wù)是把這樣的類型寫入一個輸出文件流(ofstream)中。應(yīng)用程序必須這些值存儲為相應(yīng)的二進制形式,基于這個目的,應(yīng)使用write() 和read() 成員函數(shù)。write() 以某個變量的地址和大小為參數(shù),把該變量的位模式寫入一個文件流中。read() 的兩個參數(shù)為char*和long類型,分別指示內(nèi)存緩沖區(qū)的地址和字節(jié)大小。下面的例子演示如何在ofstream中保存兩個整數(shù):

 

  1. #include <fstream>  
  2. using namespace std;  
  3. int main()  
  4. {  
  5.  int x,y; // mouse coordinates  
  6.  // ..assign values to x and y  
  7.  ofstream archive("coord.dat", ios::binary);  
  8.  archive.write(reinterpret_castchar *>(&x), sizeof (x));  
  9.  archive.write(reinterpret_castchar *>(&x), sizeof (x));  
  10.  archive.close();  

 

使用reinterpret_cast<>是必要的,因為write()的***個參數(shù)類型為const char*,但&x和&y是int*類型。

以下代碼讀取剛才存儲的值:

 

  1. #include <fstream>  
  2. using namespace std;  
  3. int main()  
  4. {  
  5.  int x,y;  
  6.  ifstream archive("coord.dat");  
  7.  archive.read((reinterpret_castchar *>(&x), sizeof(x));  
  8.  archive.read((reinterpret_castchar *>(&y), sizeof(y));  
  9. }  

序列化對象

要序列化一個完整的對象,應(yīng)把每個數(shù)據(jù)成員寫入文件中:

 

  1. class MP3_clip  
  2. {  
  3.  private:  
  4.  std::time_t date;  
  5.  std::string name;  
  6.  int bitrate;  
  7.  bool stereo;  
  8.  public:  
  9. void serialize();  
  10. void deserialize();  
  11. //..  
  12. };  
  13. void MP3_clip::serialize()  
  14. {  
  15.  int size=name.size();// store name's length  
  16.  //empty file if it already exists before writing new data  
  17.  ofstream arc("mp3.dat", ios::binary|ios::trunc);  
  18.  arc.write(reinterpret_castchar *>(&date),sizeof(date));  
  19.  arc.write(reinterpret_castchar *>(&size),sizeof(size));  
  20.  arc.write(name.c_str(), size+1); // write final '\0' too  
  21.  arc.write(reinterpret_castchar *>(&bitrate),  
  22.  sizeof(bitrate));  
  23.  arc.write(reinterpret_castchar *>(&stereo),  
  24.  sizeof(stereo));  

 

實現(xiàn)deserialize() 需要一些技巧,因為你需要為字符串分配一個臨時緩沖區(qū)。做法如下:

 

  1. void MP3_clip::deserialize()  
  2. {  
  3.  ifstream arce("mp3.dat");  
  4.  int len=0;  
  5.  char *p=0;  
  6.  arc.read(reinterpret_castchar *>(&date), sizeof(date));  
  7.  arc.read(reinterpret_castchar *>(&len), sizeof(len));  
  8.  p=new char [len+1]; // allocate temp buffer for name  
  9.  arc.read(p, len+1); // copy name to temp, including '\0'  
  10.  name=p; // copy temp to data member  
  11.  delete[] p;  
  12.  arc.read(reinterpret_castchar *>(&bitrate),  
  13.  sizeof(bitrate));  
  14.  arc.read(reinterpret_castchar *>(&stereo),  
  15.  sizeof(stereo));  

 

性能優(yōu)化

你可能會感到迷惑,為什么不把整個對象一次性轉(zhuǎn)儲到文件中,而必須對每個數(shù)據(jù)成員進行序列化呢?換句話說,難道不能用下面的方式實現(xiàn)serialize() 嗎?

 

  1. void MP3_clip::serialize()  
  2. {  
  3.  ofstream arc("mp3.dat", ios::binary|ios::trunc);  
  4.  arc.write(reinterpret_castchar *>(this),sizeof(*this));  

 

不行,不能這樣做。這種方式至少存在兩個問題。通常,當(dāng)被序列化的對象還包含其它一些對象時,你不能簡單地把該對象轉(zhuǎn)儲到一個文件中并指望以后從中重建一個有效的對象。在我們的例子中,外層對象包含一個std::string成員,一個淺拷貝(shallow copy)操作會把std::string成員歸檔,但其值是時變的,意思是說每次運行程序時都可能改變。

更糟的是,由于std::string事實上并不包含一個字符數(shù)組,而是一個指針,使用淺拷貝試圖重建原始字符串是不可能的。為克服這個問題,程序沒有序列化string對象,而是歸檔其含有的字符和長度。一般來說,指針,數(shù)組和句柄應(yīng)以相同的方式進行處理。

另一個問題設(shè)計到多態(tài)對象。每個多態(tài)對象都含有一個vtpr,即一個指向虛擬函數(shù)地址分配表的隱藏指針。vtpr的值是時變的,如果你把整個多態(tài)對象轉(zhuǎn)儲到一個文件中,然后強行把歸檔后的數(shù)據(jù)添加到一個新的對象上,則其vptr可能無效并導(dǎo)致未定義的行為。再次提醒,解決方案是只對非時變的數(shù)據(jù)成員進行序列化和反序列化。另一種方法是計算vptr的確切偏移量,在從文件重建對象時不要動它。記住,vptr的位置是與實現(xiàn)相關(guān)的,因此這樣的代碼是不可移植的。

小結(jié)

雖然C++不直接支持對象持久性,但手工實現(xiàn)它并不難,只要你遵從一些基本的準(zhǔn)則:首先把每個復(fù)合對象分解為原始數(shù)據(jù)類型,然后對這些原始數(shù)據(jù)類型進行序列化。當(dāng)序列化數(shù)據(jù)時,記住要跳過時變的值。在反序列化過程中,讀取剛才存儲的值。處理string對象、數(shù)組和句柄需要一些技巧:總是要對它們解引用,存儲它們所指向的值。記住在一個單獨的字段中存儲string或數(shù)組的大小。

希望通過以上內(nèi)容的介紹,能夠給你帶來幫助。

責(zé)任編輯:于鐵 來源: 互聯(lián)網(wǎng)
相關(guān)推薦

2009-08-06 10:27:08

C#應(yīng)用程序域

2009-08-24 13:30:50

C# Windows

2010-01-26 17:16:33

C++應(yīng)用程序

2011-06-21 10:28:17

Qt Visual C++

2010-01-28 09:44:08

C++應(yīng)用程序

2009-01-19 11:07:42

C#Web.NET

2010-01-14 11:14:47

C++應(yīng)用程序

2010-01-22 13:59:34

Visual C++應(yīng)

2010-01-25 16:41:08

C++應(yīng)用程序

2010-01-28 17:14:38

Visual C++應(yīng)

2009-07-14 11:08:42

WebRendererSwing應(yīng)用程序

2009-07-21 15:02:19

ASP.NET應(yīng)用程序

2009-09-03 17:36:13

C#創(chuàng)建Web應(yīng)用程序

2009-08-25 09:39:21

創(chuàng)建C# Window

2013-06-24 10:21:47

面向?qū)ο?/a>Web應(yīng)用JavaScript

2010-02-04 17:05:53

C++動態(tài)創(chuàng)建對象

2011-09-05 10:27:02

Sencha Touc手機應(yīng)用Android

2009-08-14 17:27:30

C#Windows應(yīng)用

2023-09-27 23:24:50

C++鏈表

2009-04-12 09:25:12

Symbian諾基亞移動OS
點贊
收藏

51CTO技術(shù)棧公眾號

主站蜘蛛池模板: 国产精品免费在线 | 日本久久一区二区三区 | 久热精品视频 | 免费观看一级毛片 | 久久久激情| 国产精品视频网 | 99久久免费精品国产男女高不卡 | 黑人精品欧美一区二区蜜桃 | 中文字幕 在线观看 | 日韩成人在线视频 | av在线一区二区三区 | 超碰在线免费 | 国产高清免费视频 | 亚洲人的av| 韩国主播午夜大尺度福利 | 日韩中文字幕第一页 | 天天操夜夜操 | 精品一区二区三区入口 | 欧美一区二区三区的 | 国产高清免费在线 | 久久一区二区三区四区 | 国产精品久久久久久久久免费软件 | 欧美黄色免费网站 | 日韩欧美精品一区 | 中文天堂在线一区 | 激情一区二区三区 | 久久专区| 美女在线视频一区二区三区 | 中国一级大黄大片 | 黄色欧美在线 | 日韩视频在线一区 | 国产精品国产亚洲精品看不卡15 | 黄网站免费在线 | 成人在线视频免费观看 | 成人在线精品 | 国产99久久精品一区二区永久免费 | 精品伊人| 最新日韩在线视频 | 亚洲国产专区 | 九色在线观看 | 91精品国产综合久久精品 |