• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>

            旅途

            如果想飛得高,就該把地平線忘掉

            在C++中創建持久對象

            持久對象(persistent objects)廣泛應用于游戲、分布式數據庫系統、多媒體以及圖形應用程序中。目前C++并不直接支持持久性(persistence)(但有一些在C++未來版本中添加持久性和反射(reflection)的建議)。持久對象可以在創建它的程序的作用域之外保持自身狀態。把對象寫入一個文件并在以后重建之,或者把對象傳送到一臺遠程機器,就是這樣的例子。對持久性的支持并不象第一眼看上去那樣簡單,同一對象的大小和內存布局在不同的平臺上可能并不相同,而不同的字節次序(byte ordering),或稱為endian-ness,使事情更加復雜化。在下文中我將討論如何實現持久性,而無須求助于DCOM和 Corba之類的第三方框架。對于小型和可移植的應用程序而言,這是一種有效并令人滿意的方案。

            ?

            序列化(serialization )基礎

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

            ?

            序列化內置數據類型

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

            ?

            #include <fstream>

            using namespace std;

            int main()

            {

            ?int x,y; // mouse coordinates

            ?// ..assign values to x and y

            ?ofstream archive("coord.dat", ios::binary);

            ?archive.write(reinterpret_cast<char *>(&x), sizeof (x));

            ?archive.write(reinterpret_cast<char *>(&x), sizeof (x));

            ?archive.close();

            }

            ?

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

            ?

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

            ?

            #include <fstream>

            using namespace std;

            int main()

            {

            ?int x,y;

            ?ifstream archive("coord.dat");

            ?archive.read((reinterpret_cast<char *>(&x), sizeof(x));

            ?archive.read((reinterpret_cast<char *>(&y), sizeof(y));

            }

            ?

            序列化對象

            要序列化一個完整的對象,應把每個數據成員寫入文件中:

            ?

            class MP3_clip

            {

            private:

            ?std::time_t date;

            ?std::string name;

            ?int bitrate;

            ?bool stereo;

            public:

            ?void serialize();

            ?void deserialize();

            ?//..

            };

            ?

            void MP3_clip::serialize()

            {

            {

            ?int size=name.size();// store name's length

            ?//empty file if it already exists before writing new data

            ?ofstream arc("mp3.dat", ios::binary|ios::trunc);

            ?arc.write(reinterpret_cast<char *>(&date),sizeof(date));

            ?arc.write(reinterpret_cast<char *>(&size),sizeof(size));

            ?arc.write(name.c_str(), size+1); // write final '\0' too

            ?arc.write(reinterpret_cast<char *>(&bitrate),

            ?sizeof(bitrate));

            ?arc.write(reinterpret_cast<char *>(&stereo),

            ?sizeof(stereo));

            }

            ?

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

            ?

            void MP3_clip::deserialize()

            {

            ?ifstream arce("mp3.dat");

            ?int len=0;

            ?char *p=0;

            ?arc.read(reinterpret_cast<char *>(&date), sizeof(date));

            ?arc.read(reinterpret_cast<char *>(&len), sizeof(len));

            ?p=new char [len+1]; // allocate temp buffer for name

            ?arc.read(p, len+1); // copy name to temp, including '\0'

            ?name=p; // copy temp to data member

            ?delete[] p;

            ?arc.read(reinterpret_cast<char *>(&bitrate),

            ?sizeof(bitrate));

            ?arc.read(reinterpret_cast<char *>(&stereo),

            ?sizeof(stereo));

            }

            ?

            性能 優化

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

            ?

            void MP3_clip::serialize()

            {

            ?ofstream arc("mp3.dat", ios::binary|ios::trunc);

            ?arc.write(reinterpret_cast<char *>(this),sizeof(*this));

            }

            ?

            不行,不能這樣做。這種方式至少存在兩個問題。通常,當被序列化的對象還包含其它一些對象時,你不能簡單地把該對象轉儲到一個文件中并指望以后從中重建一個有效的對象。在我們的例子中,外層對象包含一個std::string成員,一個淺拷貝(tips/tipview.

            ?

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

            ?

            小結

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

            ?

            posted on 2007-07-31 13:01 旅途 閱讀(129) 評論(0)  編輯 收藏 引用 所屬分類: C/C++

            午夜欧美精品久久久久久久| 亚洲国产精品久久久久网站| 国内精品欧美久久精品| 人妻系列无码专区久久五月天| 日本久久久久亚洲中字幕 | 久久综合丁香激情久久| 亚洲国产成人久久综合一区77| 久久精品人人做人人妻人人玩| 无码国内精品久久综合88 | 亚洲中文字幕无码久久精品1 | 国产成人精品久久| 国产一区二区三区久久精品| 国产精品99久久久久久宅男小说| 久久99久久99精品免视看动漫| 亚洲国产精品成人久久| 欧美粉嫩小泬久久久久久久| 国产精自产拍久久久久久蜜| yellow中文字幕久久网| 久久久久高潮毛片免费全部播放| 欧美精品一区二区久久| 久久夜色tv网站| 国产成人久久精品区一区二区| 精品久久久久久久久中文字幕| 亚洲欧美精品一区久久中文字幕| 夜夜亚洲天天久久| 精品一区二区久久久久久久网站| 人妻精品久久久久中文字幕一冢本 | 亚洲国产香蕉人人爽成AV片久久 | 亚洲成色www久久网站夜月| 亚洲精品久久久www| 久久综合久久性久99毛片| 久久99精品九九九久久婷婷| 99精品久久久久久久婷婷| 久久99精品国产99久久6男男| 久久久久综合网久久| 7国产欧美日韩综合天堂中文久久久久 | 久久精品9988| 久久精品国产99久久丝袜| 亚洲一级Av无码毛片久久精品| 久久久精品日本一区二区三区| 久久亚洲中文字幕精品一区四|