内存数据库机制的使用研究报告
相对于传统磁盘数据库,内存数据库通过将数据完全加载到内存,在内存中实现对数据的管理,在数据同步、数据传送、事务处理、并行操作等方面进行了相应的改进设计,使得内存数据库在处理数据上能够比磁盘数据库快得多,可以有效地解决计费系统中信控、实时累账等部分对系统响应要求高的问题。
项目决定在信控模块使用内存数据库机制取代账单中心模式进行实时累账优惠触发信控。
以下是对FastDB进行的几点说明:
【开源代码链接:http://sourceforge.net/projects/fastdb】
1、
功能归纳
1
)
内存数据库具备传统数据库的一些基本功能:
A
:
数据的管理,内存数据库机制是支持永久数据的管理的,包括数据库的的定义、存储、维护等功能。
B
:
数据的操作,内存数据库支持对数据进行增,删,改,查,数据完整性校验等一些基本功能。
C
:
事务管理,内存数据库支持调度,进程间、线程间的一些并发等操作。
d.
数据恢复备份机制,内存数据库支持在线备份和系统崩溃后的自动恢复。
2
)
FastDB
在功能上特有的一些特点
A
:
查询优化,与传统数据库管理系统咋红执行查询相比,当所有的数据存在于内存当中时,查询的数度是非常快的,但是
FastDB
还是进一步使用了索引
(hash,T-tree)
、反向指针和查询并行化进行了查询优化。
B
:
提供了一个灵活方便的应用程序语言接口,能够方便写出查询等语句。
2
、
工作原理
FastDB
是一个高效率的内存数据库系统,在磁盘上的数据库文件和使用该数据库的每一个应用程序占用的虚拟内存空间相映射,这样取消了数据文件和缓冲池中的数据传输。再将整个文件数据读入内存,并且使用了高性能的锁工具实现了只读模式线程间、单个更改模式线程和多个只读模式线程间的并发执行。
FastDB
通过位图实现对内存进行分配,最小单位块是分配量子(
16
字节)。如此大大提高了数据引用的局部性(对象数据尽可能分配在连续的内存区域),最小化了修改页的数目和减少了事务提交时间。事务提交协议基于一个影子根页算法,对数据库执行原子更新操作,恢复效率很高,在存储数据结构上可以采用
T-tree
结构(
T-tree
和
AVL-tree
相似,只是
T-tree
中每个节点中顺序存储了多个值),对于大量相似重复性数据的查询性能相当高;也可以采用
Hash
存储,这是用关键字段定位表中记录的最好办法(采用等号进行查询)。
影子根页算法概述:
FastDB
数据库中每条对象都具有唯一的标识符(
OID
),用作一个数组(对象索引)的下标,元素值表示对象的一个句柄,在
FastDB
数据库中存在两个索引(当前索引和影子索引),当某个对象第一次被修改时,它会创建一个副本,当前索引中的对象句柄被修改指向副本,影子索引仍然包含一个指向该对象原始版本的句柄。所有更改发生在副本上,
FastDB
在对象索引的一个特殊位图页上标记出哪个索引包含修改过的对象句柄。
当一个事务被提交时,
FastDB
首先检查对象索引的尺寸的大小,若增长了,还会重新为对象索引的影子副本重新分配内存,然后释放“旧对象”占用的内存,释放后,将修改过的所有位图页
flush
到磁盘上,然后
FastDB
将改变数据库头部中的当前对象索引指示符,以切换对象索引的角色。当前对象索引将变成影子索引之后,
FastDB
把修改过的所有句柄从新的对象索引中复制到先前是影子的、现在已成为当前的对象索引中。此时,两个索引都得到了同步。(具体内存块的移动索引管理细节需要进一步琢磨)
3、
部署方法
1
)
应用程序编译环境需求,首先是任何一个
FastDB
应用程序必须包含头文件:
fastdb.h
;
然后是可以选择调用库文件(
FastDB
编译后提供静态库
(libfastdb_r.a)
和共享库两种库
(libfastdb_r.so/ libfastdb_r.so.2)
给调用);
最后是
FastDB
提供很多编译选项接口,用户可以根据需要进行设置,比如:容错支持,无盘模式,锁检测清理机制等等功能。
2
)
运行系统环境需求,理论上说,内存加载的数据库文件规模最小是
1MB
,上限就是内存和磁盘的容量了(
FastDB
的整个优化设计是基于真个数据库系统存放在机器物理内存中,但是它依然支持将应用在规模超过物理内存的数据库上,只是效率不会很高)
4
、
接口调用方法
1
)
FastDB
提供了
subsql
交互式工具供用户进行备份,查询,监控等。(按着提示走即可)
2
)
C++
接口
A
:
打开或创建数据库:
dbDatabase db(parameter);
db.open(parameter);
mode
的有:
dbReadOnly
,
dbAllAccess
,
dbConcurrentRead
,
dbConcurrentUpdate
四模式
B
:
FastDB
支持的数据类型:
类型
描述
bool
布尔类型
(true,false)
int1
一个字节的带符号整型
(-128..127)
int2
两个字节的带符号整型
(-32768..32767)
int4
四个字节的带符号整型
(-2147483648..2147483647)
int8
八个字节的带符号整型
(-2**63..2**63-1)
real4
四个字节的
ANSI
浮点型
real8
八个字节的双精度浮点型
char const*
非中断整型
dbReference<T>
到类
T
的指针。
dbArray<T>
元素类型是
T
的动态数组。
C
:
FastDB
对表的接口描述
C++
需要用类的形式来定义表结构,然后一一映射到表的
fields
,如果类有方法就得用宏:
CLASS_DESCRIPTOR(name,field_list)
,进行描述,还有方法宏
TYPE_DESCRIPTOR(field_list)
(具体的查看的帮助文档)
最后只需要将类进行注册:
REGISTER(Class Name);
D
:
游标
游标有两种模式:
dbCursorViewOnly
,
dbCursorForUpdate
定义举例
dbCursor<Class Name> instance (dbCursorForUpdate);
提供了数据库的改、删、查方法接口
Instance.udate()
Instance.remove()/
removeAllSelected
() /
removeAll
()
Instance.select()/select(dbquery &q);
E
:
dbquery
Q;
Q = “id = ” idvalue,”and datetime = ”,currtime;
F
:
插
数据的
insert
方法
FastDB
提供了重载
dbReference<T> insert(T const& record);
所以使用起来是很简单的。
具体细节用法查阅手册。
Hoho
。。。
5
、
改造计划
暂时不用改造现有机制,如果不想在内存数据库当中形成综合账单表(从原来的账单中心的四张内存表(累账账单表,周期性费用表,优惠结果表,调账结果表)中提取),减少冗余,就需要改造其现有的事务提交机制。使得事务提交后,程序能够控制数据改变对其他进程的可见性。
6
、本人的两点担心
1
)
FastDB
适合主导读取模式的应用程序,在大规模数据群处理上,进行更改事务处理的效率和准确性能上有待后查。
2
)
FastDB
虽然是一个极其优秀的开源内存数据库,但毕竟没有或者很少经过商用的专业的测试。系统中复杂的内存管理过程,可能容易产生过多的内存碎片,导致系统的不稳定。