sqlite内存数据库和文件数据库的同步[转]
由于sqlite对多进程操作支持效果不太理想,在项目中,为了避免频繁读写 文件数据库带来的性能损耗,我们可以采用操作sqlite内存数据库,并将内存数据库定时同步到文件数据库中的方法。
实现思路如下:
1、创建文件数据库;
2、创建内存数据库(文件数据库、内存数据库的内幕表结构需要一致);
3、在内存数据库中attach文件数据库,这样可以保证文件数据库中的内容在内存数据库中可见;
4、对于insert、select操作,在内存数据库中完成,对于delete、update操作,需要同时访问内存、文件数据库;
5、定时将内存数据库中的内容flush到文件数据库。
通过sqlite的cAPI实现代码如下:
- const char* file_database_path = "/home/tom/test/database/filedb"; //文件数据库存放路径
- const char* sql_create_data = "CREATE TABLE testinfo (id TEXT PRIMARY KEY, message TEXT, offset INTEGER, timestamp INTEGER);";
- const char* sql_insert_data = "INSERT OR REPLACE INTO MAIN.testinfo VALUES('%s', '%s', %d, %d);";
- const char* sql_delete_data = "DELETE FROM MAIN.testinfo WHERE id = '%s'; DELETE FROM filedb.testinfo WHERE id = '%s';"; //删除数据库,需同时删除内存、文件数据库中的内容
- const char* sql_update_data = "UPDATE MAIN.testinfo SET message = '%s', offset = %d, timestamp = %d where id = '%s'; UPDATE filedb.testinfo SET message = '%s', offset = %d, timestamp = %d where id = '%s';";//更新数据库,需同时更新内存、文件数据库中的内容
- const char* sql_search_data = "SELECT * FROM MAIN.testinfo WHERE timestamp BETWEEN %d AND %d union SELECT * FROM testdb.testinfo WHERE timestamp BETWEEN %d AND %d;"; //查找数据库,将内存、文件数据库中查找出的内容合并
- const char* sql_transfer_data = "INSERT OR REPLACE INTO filedb.testinfo SELECT * FROM testinfo;"; //将内存数据库中的信息同步到文件数据库中
- const char* sql_delete_memory_table = "DELETE FROM testinfo;"; //内存数据库中的内容同步结束后,清空
- int InsertRecord(DATA_TYPE type, const char* id, const char* message, int offset, int timestamp)
- {
- int rc = 0;
- char* errMsg = NULL;
- char sqlcmd[512] = {0};
- time_t insertTimestamp = 0;
- snprintf(sqlcmd, sizeof(sqlcmd), sql_insert_data, id, message, offset, timestamp);
- rc = sqlite3_exec(memdb, sqlcmd, NULL, NULL, &errMsg);
- if (SQLITE_OK != rc) {
- fprintf(stderr, "cat't add record to memory database %s, sqlcmd=%s, err:%s\n", map_data_table[type].data_table_name, sqlcmd, errMsg);
- return -1;
- }
- return 0;
- }
- int UpdateRecord(DATA_TYPE type, const char* id, const char* message, int offset, int timestamp)
- {
- int rc = 0;
- char* errMsg = NULL;
- char sqlCmd[512] = {0};
- snprintf(sqlCmd, sizeof(sqlCmd), sql_update_data, message, offset, timestamp, id, message, offset, timestamp, id);
- rc = sqlite3_exec(memdb, sqlCmd, NULL, NULL, &errMsg);
- if (SQLITE_OK != rc) {
- fprintf(stderr, "cat't update record %s:%s\n", map_data_table[type].data_table_name, errMsg);
- return -1;
- }
- return 0;
- }
- int DeleteRecord(DATA_TYPE type, const char* id)
- {
- int rc = 0;
- char* errMsg = NULL;
- char sqlcmd[512] = {0};
- snprintf(sqlcmd, sizeof(sqlcmd), sql_delete_data, id, id);
- rc = sqlite3_exec(memdb, sqlcmd, NULL, NULL, &errMsg);
- if (SQLITE_OK != rc) {
- fprintf(stderr, "cat't delete record %s:%s\n", map_data_table[type].data_table_name, errMsg);
- return -1;
- }
- return 0;
- }
- int QueryMessage(DATA_TYPE type, int startTime, int endTime)
- {
- int rc = 0;
- char *errMsg = NULL;
- sqlite3 *filedb = NULL;
- char** pRecord = NULL;
- int row = 0;
- int column = 0;
- char sqlcmd[512] = {0};
- if (type > VEP_NELEMS(map_data_table) || type < 0) {
- return -1;
- }
- rc = sqlite3_open(file_database_path, &filedb);
- if (SQLITE_OK != rc) {
- fprintf(stderr, "cat't open database:%s\n", sqlite3_errmsg(filedb));
- sqlite3_close(filedb);
- return -1;
- }
- snprintf(sqlcmd, sizeof(sqlcmd), sql_search_data, startTime, endTime, startTime, endTime);
- rc = sqlite3_get_table(filedb, sqlcmd, &pRecord, &row, &column, &errMsg);
- if (SQLITE_OK != rc) {
- fprintf(stderr, "cat't get table from%s:%s\n", map_data_table[type].data_table_name, errMsg);
- return -1;
- }
- int i;
- printf("row = %d, column = %d\n", row, column);
- for(i = 0; i < 2*column; i++)
- {
- printf("%s ", pRecord[i]);
- }
- printf("\n");
- return 0;
- }
- //定时调用此函数将内存数据中的内容同步到文件数据库
- int Flush(){
- int i = 0;
- int rc = 0;
- char* errMsg = NULL;
- char sqlcmd[512] = {0};
- snprintf(sqlcmd, sizeof(sqlcmd), sql_transfer_data);
- rc = sqlite3_exec(memdb, sqlcmd, NULL, NULL, &errMsg);
- if (SQLITE_OK != rc) {
- fprintf(stderr, "cat't transfer memory database %s to file databasede:%s\n", map_data_table[i].data_table_name, sqlite3_errmsg(memdb));
- sqlite3_close(memdb);
- return -1;
- }
- snprintf(sqlcmd, sizeof(sqlcmd), sql_delete_memory_table);
- rc = sqlite3_exec(memdb, sqlcmd, NULL, NULL, &errMsg);
- return 0;
- }
- //创建文件数据库
- int CreateDbOnFile()
- {
- sqlite3 *db = NULL;
- int rc = 0;
- char* errMsg = NULL;
- char sqlcmd[512] = {0};
- int i = 0;
- rc = sqlite3_open(file_database_path, &db);
- if (SQLITE_OK != rc) {
- fprintf(stderr, "cat't open database:%s\n", sqlite3_errmsg(db));
- sqlite3_close(db);
- return -1;
- }
- snprintf(sqlcmd, sizeof(sqlcmd), sql_create_data);
- rc = sqlite3_exec(db, sqlcmd, NULL, NULL, &errMsg);
- if (SQLITE_OK != rc) {
- fprintf(stderr, "cat't create file database testinfo:%s\n", errMsg);
- sqlite3_close(db);
- return -1;
- }
- sqlite3_close(db);
- return 0;
- }
- //创建内存数据库
- int CreateDbOnMemery()
- {
- int rc = 0;
- char* errMsg = NULL;
- char sqlcmd[512] = {0};
- int i = 0;
- rc = sqlite3_open(":memory:", &memdb);
- if (SQLITE_OK != rc) {
- fprintf(stderr, "cat't open database:%s\n", sqlite3_errmsg(memdb));
- sqlite3_close(memdb);
- return -1;
- }
- snprintf(sqlcmd, sizeof(sqlcmd), sql_create_data);
- rc = sqlite3_exec(memdb, sqlcmd, NULL, NULL, &errMsg);
- if (SQLITE_OK != rc) {
- fprintf(stderr, "cat't create memory database %s\n", errMsg);
- sqlite3_close(memdb);
- return -1;
- }
- return 0;
- }
- //解绑数据库
- int DetachDb()
- {
- int rc = 0;
- char* errMsg = NULL;
- char sqlcmd[512] = {0};
- snprintf(sqlcmd, sizeof(sqlcmd), "DETACH '%s'", "filedb");
- rc = sqlite3_exec(memdb, sqlcmd, NULL, NULL, &errMsg);
- if (SQLITE_OK != rc) {
- fprintf(stderr, "detach file database failed:%s:%s\n", file_database_path, errMsg);
- sqlite3_close(memdb);
- return -1;
- }
- return 0;
- }
- //将文件数据库作为内存数据库的附加数据库
- int AttachDb()
- {
- int rc = 0;
- char* errMsg = NULL;
- char sqlcmd[512] = {0};
- snprintf(sqlcmd, sizeof(sqlcmd), "ATTACH '%s' AS %s", file_database_path, "filedb");
- rc = sqlite3_exec(memdb, sqlcmd, NULL, NULL, &errMsg);
- if (SQLITE_OK != rc) {
- fprintf(stderr, "cat't attach database %s:%s\n", file_database_path, errMsg);
- sqlite3_close(memdb);
- return -1;
- }
- return 0;
- }
- //初始化数据库,分别创建文件数据库、内存数据库并把文件数据库attach到内存数据库上
- int InitSqliteDb()
- {
- int retval = 0;
- retval = CreateDbOnFile();
- if (retval != 0) {
- return retval;
- }
- retval = CreateDbOnMemery();
- if (retval != 0) {
- return retval;
- }
- retval = AttachDb();
- if (retval != 0) {
- return retval;
- }
- return 0;
- }
sqlite内存数据库和文件数据库的同步[转]的更多相关文章
- FIREDAC操作SQLITE内存数据库
SQLite不仅可以把数据库放在硬盘上,还可以放在内存中,经测试,同样条件下数据库放在内存中比放在硬盘上插入记录速度快差不多3倍. 但数据库放在内存中时有如下缺陷: 1.断电或程序崩溃后数据库就会消失 ...
- DBImport v3.5 中文版发布:数据库定时同步及文档生成工具(IT人员必备)
前言: 趁着最近的休息时间,只能多勤快些:多写代码,多更新文章. 因为一旦投入新的工作,估计博客又会恢复到一年才产几篇的状态. 对于DBImport,因为用户的意见,增加了一个亮点功能,让软件B格升为 ...
- SymmetricDS 数据库双向同步开源软件入门
一句话概括该软件:SymmetricDS是一个文件和数据库同步软件,开源的,支持多主复制,同步时过滤和在异构的网络环境中进行数据转换传输.它支持单向和双向上的多个订阅者,异步的数据复制. 以下是从CS ...
- SQLite内存数据库
[转]SQLite内存数据库 http://www.cnblogs.com/liuyong/archive/2010/09/14/1826152.html SQLite 介绍 一. SQLite 是实 ...
- 【原创】System.Data.SQLite内存数据库模式
对于很多嵌入式数据库来说都有对于的内存数据库模式,SQLite也不例外.内存数据库常常用于极速.实时的场景,一个很好的应用的场景是富客户端的缓存数据,一般富客户端的缓存常常需要分为落地和非落地两种,而 ...
- mysql数据库主从同步
环境: Mater: CentOS7.1 5.5.52-MariaDB 192.168.108.133 Slave: CentOS7.1 5.5.52-MariaDB 192.168. ...
- .NET平台开源项目速览(3)小巧轻量级NoSQL文件数据库LiteDB
今天给大家介绍一个不错的小巧轻量级的NoSQL文件数据库LiteDB.本博客在2013年也介绍过2款.NET平台的开源数据库: 1.[原创]开源.NET下的XML数据库介绍及入门 2.[原创]C#开源 ...
- mysql主从数据库不同步的2种解决方法(转)
今天发现Mysql的主从数据库没有同步 先上Master库: mysql>show processlist; 查看下进程是否Sleep太多.发现很正常. show master status; ...
- Mysql的主从数据库没有同步的解决办法
Mysql的主从数据库没有同步的解决办法 今天发现Mysql的主从数据库没有同步 先上Master库: mysql>show processlist; 查看下进程是否Sleep太多.发现很正常. ...
随机推荐
- JavaScirpt(JS)——DOM文档对象模型
一.HTML DOM介绍 HTML DOM 是 W3C 标准(是 HTML 文档对象模型的英文缩写,Document Object Model for HTML). HTML DOM 定义了用于 HT ...
- 详解JavaScript UTC时间转换方法
这篇文章主要介绍了JavaScript UTC时间转换方法,介绍了本地时间到UTC时间的转换.UTC日期到本地日期的转换,感兴趣的小伙伴们可以参考一下 一.前言 1.UTC: Universal Ti ...
- Apache Flume
An Event is a unit of data that flows through a Flume agent. The Event flows from Source to Channel ...
- 多线程(五)~ wait/notify机制(等待/通知)
首先我们来看一张图,这张图描述了线程操作的几个步骤. 图已经描述的很清楚了,这里除了wait()之外,其他的前面都已经接触过了. 这一章我们主要来说一下和wait()相关的操作,其实和wait()相关 ...
- JS教程之实现加载图片时百分比进度
思路:思路其实很简单,ajax执行时,会生成一个event对象,其中会包含要加载的文件的大小和当前已经加载完成部分的大小,通过这两个值即可计算出百分比 事件介绍onprogress 当浏览器正在加载媒 ...
- Third week-homework(员工管理系统)
需求: 可以查询员工所有信息 可以修改员工信息 可以增加新员工 code: import sys,json # yuangong = { # "1": ["faker&q ...
- 关于Oracle RAC节点间免密码策略
在RAC安装过程中无论grid或是Oracle软件都需要配置节点间免密码策略.配置方法网上有很多,oracle和可以帮你配置,此处不多说. 但是在RAC运行过程中是不需要免密码登陆的,所以在集群安装完 ...
- SAP S4CRM 1811 服务订单API介绍
Jerry在今年2月28日,SAP Customer Management for S/4HANA 1.0正式问世这个具有纪念意义的日子,同时发布了中英文版的博客进行介绍. 英文版发在SAP社区上,至 ...
- IOS Runtime的用法
什么是runtime? 1> runtime是一套底层的C语言API(包含很多强大实用的C语言数据类型.C语言函数)2> 实际上,平时我们编写的OC代码,底层都是基于runtime实现的* ...
- http中COOKIE和SESSION有什么区别?(转知乎)
作者:知乎用户链接:https://www.zhihu.com/question/19786827/answer/28752144来源:知乎著作权归作者所有.商业转载请联系作者获得授权,非商业转载请注 ...