浅谈C++应用程序中创建持久化对象

  持久化的对象,是已经存储到数据库或保存到本地硬盘中的对象,我们称之为持久化对象。下面介绍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中保存两个整数:


  使用reinterpret_cast<>是必要的,因为write()的第一个参数类型为const char*,但&x和&y是int*类型。

  以下代码读取刚才存储的值:


  序列化对象

  要序列化一个完整的对象,应把每个数据成员写入文件中:

  实现deserialize() 需要一些技巧,因为你需要为字符串分配一个临时缓冲区。做法如下:


  性能优化

  你可能会感到迷惑,为什么不把整个对象一次性转储到文件中,而必须对每个数据成员进行序列化呢?换句话说,难道不能用下面的方式实现serialize() 吗?


  不行,不能这样做。这种方式至少存在两个问题。通常,当被序列化的对象还包含其它一些对象时,你不能简单地把该对象转储到一个文件中并指望以后从中重建一个有效的对象。在我们的例子中,外层对象包含一个std::string成员,一个浅拷贝(shallow copy)操作会把std::string成员归档,但其值是时变的,意思是说每次运行程序时都可能改变。

  更糟的是,由于std::string事实上并不包含一个字符数组,而是一个指针,使用浅拷贝试图重建原始字符串是不可能的。为克服这个问题,程序没有序列化string对象,而是归档其含有的字符和长度。一般来说,指针,数组和句柄应以相同的方式进行处理。

  另一个问题设计到多态对象。每个多态对象都含有一个vtpr,即一个指向虚拟函数地址分配表的隐藏指针。vtpr的值是时变的,如果你把整个多态对象转储到一个文件中,然后强行把归档后的数据添加到一个新的对象上,则其vptr可能无效并导致未定义的行为。再次提醒,解决方案是只对非时变的数据成员进行序列化和反序列化。另一种方法是计算vptr的确切偏移量,在从文件重建对象时不要动它。记住,vptr的位置是与实现相关的,因此这样的代码是不可移植的。

  小结

  虽然C++不直接支持对象持久性,但手工实现它并不难,只要你遵从一些基本的准则:首先把每个复合对象分解为原始数据类型,然后对这些原始数据类型进行序列化。当序列化数据时,记住要跳过时变的值。在反序列化过程中,读取刚才存储的值。处理string对象、数组和句柄需要一些技巧:总是要对它们解引用,存储它们所指向的值。记住在一个单独的字段中存储string或数组的大小,希望通过以上内容的介绍,能够给你带来帮助。

时间: 2024-08-30 16:37:01

浅谈C++应用程序中创建持久化对象的相关文章

浅谈在JAVA项目中LOG4J的使用_java

一.直接使用: //输出到项目文件夹下output1.txt文件中 ////////////////////////////// // DEBUG - Here is some DEBUG // INFO - Here is some INFO // WARN - Here is some WARN // ERROR - Here is some ERROR // FATAL - Here is some FATAL ////////////////////////////// package

为什么c程序中创建了数据文件,再次打开源代码运行时数据没了?

问题描述 为什么c程序中创建了数据文件,再次打开源代码运行时数据没了? 为什么c程序中创建了数据文件,再次打开源代码运行时数据没了? http://blog.csdn.net/hackbuteer1/article/details/6573488# 就是这个通讯录的代码,在里面创建通讯录后,再次打开运行显示通讯录时就提示通讯录为空. 解决方案 写入文件是否错误,用winhex或者ultraedit看看写文件本身有没有写对. 如果没有,就是写的问题,如果对的,那么就是读的问题.再具体调试有问题的程

浅谈js在html中的加载执行顺序,多个jquery ready执行顺序_javascript技巧

jQuery $(document).ready()执行顺序: 当页面DOM 元素全部加载完毕后就执行.ready().$(document).ready()是DOM结构绘制完毕后就执行,不必等到加载完毕. 如果在.ready()执行之前有javascript代码存在,那么javascript将怎么执行呢? 答案是先执行.ready()之前的javascript代码,然后执行.ready(). 多个$(document).ready()的执行顺序并非单纯的顺序执行,其与嵌套层级也有一定的关系.

浅谈redis在项目中的应用_Redis

redis在项目中的应用 ps:PHP 会自动 关redis连接 不需要手动关 对于临时的数据 可以不经过数据库直接redis上操作 /*消息队列实例 */ public function insertinfo(){ //连接本地的 Redis 服务 $redis = new \Redis(); $redis->connect('127.0.0.1', 6379); //存储数据到列表中 $infos = array('info1' => 66, 'info2' => 88); $red

浅谈PHP检查数组中是否存在某个值 in_array 函数_php实例

PHP in_array() 函数检查数组中是否存在某个值,如果存在则返回 TRUE ,否则返回 FALSE . 语法: bool in_array( mixed needle, array array [, bool strict] ) 参数说明: 参数 说明 needle 需要在数组中搜索的值,如果是字符串,则区分大小写 array 需要检索的数组 strict 可选,如果设置为 TRUE ,则还会对 needle 与 array 中的值类型进行检查 例子: <?php $arr_a = a

浅谈Java的String中的subString()方法_java

方法如下: public String substring(int beginIndex, int endIndex) 第一个int为开始的索引,对应String数字中的开始位置, 第二个是截止的索引位置,对应String中的结束位置 1.取得的字符串长度为:endIndex - beginIndex; 2.从beginIndex开始取,到endIndex结束,从0开始数,其中不包括endIndex位置的字符 如: "hamburger".substring(4, 8) returns

浅谈s:select 标签中list存放map对象的使用_java

1.XXXAction.java private List<Map<String, String>> maptest = null; public List<Map<String, String>> getMaptest() { return maptest; } public void setMaptest(List<Map<String, String>> maptest) { this.maptest = maptest; }

浅谈PHP检查数组中是否存在某个值 in_array 函数

PHP in_array() 函数检查数组中是否存在某个值,如果存在则返回 TRUE ,否则返回 FALSE . 语法: bool in_array( mixed needle, array array [, bool strict] ) 参数说明: 参数 说明 needle 需要在数组中搜索的值,如果是字符串,则区分大小写 array 需要检索的数组 strict 可选,如果设置为 TRUE ,则还会对 needle 与 array 中的值类型进行检查 例子: <?php $arr_a = a

使用Transact-SQL在SQL Server中创建数据库对象

&http://www.aliyun.com/zixun/aggregation/37954.html">nbsp;   Adventure Works Cycles 的数据仓库中有几个非常大的表.为改善此数据的存储,您决定对几个跨文件组的表进行分区.这样您就可以控制基础磁盘存储. 先决条件 在开始此实验之前,您必须: 使用 Transact-SQL 在 Microsoft® SQL Server® 数据库中创建数据库对象的经验. 实验设置 实验场景 Adventure Works