SQLiteC/C++接口详细介绍之sqlite3类(十八)

返回目录:SQLite—免费开源数据库系列文章目录  

上一篇:SQLiteC/C++接口详细介绍之sqlite3类(十七)

下一篇:SQLiteC/C++接口详细介绍sqlite3_stmt类(一)

56.sqlite3_update_hook

函数功能:向应用程序通知每一个 INSERT、UPDATE 或 DELETE 语句。

当INSERT、UPDATE或DELETE语句执行时,SQLite3将检测表或触发器并执行任何相关的操作。最后,SQLite3将调用与sqlite3_update_hook关联的回调函数以通知应用程序。

sqlite3_update_hook函数原型如下:

void sqlite3_update_hook(
  sqlite3 *db,
  void (*callback)(void *, int, char const *, char const *, sqlite3_int64),
  void *arg
);

参数说明:

- db:一个由sqlite3_open或sqlite3_open_v2返回的数据库句柄。
- callback:一个指向回调函数的指针。此回调函数由SQLite3调用以通知应用程序每次执行了INSERT、UPDATE或DELETE语句。该回调函数的声明应该匹配如下:

  void (*callback)(void *, int, char const *, char const *, sqlite3_int64)

  回调函数的第一个参数为表达了被改变的各种值,是由用户提供的Void指针arg决定的 。
  第二个参数是SQL语句的类型,即SQLITE_INSERT、SQLITE_UPDATE或SQLITE_DELETE。
  第三个和第四个参数表明哪个数据库和表被修改了。
  第五个参数是被修改记录的行ID。

- arg:在调用回调函数时,作为其中一个参数把它传递回去的任意类的指针。

下面是一个示例程序,演示了如何使用sqlite3_update_hook函数:

#include <sqlite3.h>
#include <stdio.h>
static void update_callback(
    void *pArg,
    int iCmd,
    char const *zDb,
    char const *zTableName,
    sqlite3_int64 rowid
) {
    printf("UPDATE hook: %s %s id=%lld\n", zDb, zTableName, rowid);
}
int main() {
    sqlite3 *db;
    sqlite3_open_v2(":memory:", &db, SQLITE_OPEN_CREATE | SQLITE_OPEN_READWRITE, NULL);
    sqlite3_exec(db, "CREATE TABLE test (id INTEGER PRIMARY KEY, value TEXT)", NULL, NULL, NULL);
    sqlite3_update_hook(db, update_callback, NULL);
    sqlite3_exec(db, "INSERT INTO test (id, value) VALUES (1, 'hello')", NULL, NULL, NULL);
    sqlite3_exec(db, "UPDATE test SET value = 'world' WHERE id = 1", NULL, NULL, NULL);
    sqlite3_exec(db, "DELETE FROM test WHERE id = 1", NULL, NULL, NULL);
    sqlite3_close(db);
    return 0;
}

该示例中,我们创建了一个名为“test”的表,然后在sqlite3_update_hook中注册了一个回调函数。我们随后在表格中插入一行,更新这一行的记录,再将这一行的记录删除。在每一次执行修改操作时,SQLite3都会调用我们注册的回调函数,它将记录当前执行的操作并输出相关信息。

程序输出的结果为:

UPDATE hook: main test id=1
UPDATE hook: main test id=1
UPDATE hook: main test id=1
可以看到,程序成功使用sqlite3_update_hook函数注册了一个回调函数,然后在每一次执行修改操作时,都会输出钩子的相应信息。

57.sqlite3_wal_autocheckpoint

函数功能:自动调度WAL模式下的日志文件检查点。

在WAL模式下,混合使用事务和独立表锁,事务使用WAL日志文件,在 WAL 模式下,删除语句不再真正执行删除,而仅在 WAL 日志文件中形成删除记录。WAL记录将在另外的线程或进程读取和删除。当WAL日志文件变得非常大时,它将占用大量磁盘空间,这将影响性能。

SQLite3允许通过自动检查点机制来控制WAL日志文件的增长。当应用程序不再使用WAL日志文件时,可以让SQLite3自动检查点。在自动检查点机制的帮助下,SQLite3将创建一个新的数据库文件,将WAL日志文件中的相关内容复制到新的数据库文件中,并将WAL日志文件重命名为删除状态。由于复制数据到新文件和删除旧的WAL日志文件都是涉及磁盘I / O的操作,因此要小心进行自动检查点机制的配置。

sqlite3_wal_autocheckpoint函数用于设置WAL文件的自动检查点算法。它将一个整数值设置为一个逻辑步骤数。在WAL模式下,每当写日志文件达到nSize个逻辑步骤时,将自动运行检查点。

sqlite3_wal_autocheckpoint函数原型如下:

int sqlite3_wal_autocheckpoint(sqlite3 *db, int N);

参数说明:

- db:由sqlite3_open或sqlite3_open_v2打开的数据库连接句柄。
- N:表示WAL日志文件中的逻辑步骤数。

返回值:设置和清除自动检查点提醒程序的SQLITE_OK,或一个错误代码。

下面是一个示例程序,演示了如何使用sqlite3_wal_autocheckpoint函数。该程序使用WAL模式编写“test”表中的几个条目,然后在自动检查点机制的帮助下设置检查点。在程序退出之前,它还回滚了最后一个此插入操作。注意,设置的逻辑步骤数只是一个量级,实际值取决于数据库的大小和预计的WAL日志文件大小。

#include <sqlite3.h>
#include <stdio.h>
int main() {
    sqlite3 *db;
    char *zErrMsg = 0;
    sqlite3_open_v2(":memory:", &db, SQLITE_OPEN_CREATE | SQLITE_OPEN_READWRITE | SQLITE_OPEN_URI, NULL);
    sqlite3_exec(db, "PRAGMA journal_mode=WAL;", NULL, NULL, NULL);
    sqlite3_exec(db, "CREATE TABLE test (id INTEGER PRIMARY KEY, value TEXT);", NULL, NULL, &zErrMsg);
    char *sql_insert = "INSERT INTO test VALUES (null, 'Hello World!');";
    sqlite3_stmt *stmt;
    sqlite3_prepare_v2(db, sql_insert, -1, &stmt, NULL);
    for (int i = 0; i < 100; i++) {
        sqlite3_step(stmt);
        sqlite3_reset(stmt);
        if (i == 50) {
            sqlite3_wal_checkpoint_v2(db, NULL, SQLITE_CHECKPOINT_TRUNCATE, NULL, NULL);
            printf("Automatically set checkpoint\n");
        }
    }
    sqlite3_finalize(stmt);
    sqlite3_exec(db, "ROLLBACK TO savepoint1;", NULL, NULL, &zErrMsg);
    printf("Rolled back transaction\n");
    sqlite3_close(db);
    return 0;
}

以上程序会创建一个名为“test”的表,使用WAL模式编写数百个条目。然后,当编写WAL日志时,程序将设置一个自动检查点,该检查点会在30个事务步骤后运行。有趣的是,我们将会回滚最后一次插入操作。在自动检查点触发之前,程序将移动到"savepoint1"保存点。

程序输出的结果如下:

Automatically set checkpoint
Rolled back transaction
可以看到,在程序的自动检查点算法处顺利地运行。执行WAL日志会在某些时候具有显著的性能和稳定性优势。如果在许多并发连接到相同的数据库时,可以使用WAL日志来避免数据库连接争用的情况。

58.sqlite3_wal_checkpoint与sqlite3_wal_checkpoint_v2

函数功能:在WAL模式下手动调用日志文件检查点。这些函数在WAL模式时可以调用。调用这些函数不是必需的,因为SQLite3会定期自动运行检查点以降低WAL日志文件的大小。但是,如果您的应用程序将使用WAL模式长时间运行,建议使用这些函数。手动调用日志文件检查点可以确保日志文件保持足够小,从而提高应用程序的性能。

sqlite3_wal_checkpoint函数适用于SQLite3 version 3.7.4 至 3.7.14.2 和 3.8.0 至 3.8.1之间的版本。sqlite3_wal_checkpoint_v2函数是3.7.14.2版本之后的版本引入的。两个函数在功能上非常相似,都可以用于手动触发WAL检查点,但sqlite3_wal_checkpoint_v2中加入了一些新的标志供选择使用。

sqlite3_wal_checkpoint_v2函数原型如下:

int sqlite3_wal_checkpoint_v2(
  sqlite3 *db,
  const char *zDb,
  int eMode,
  int *pnLog,
  int *pnCkpt
);

参数说明:

- db:由sqlite3_open或sqlite3_open_v2函数打开的一个数据库连接实例。
- zDb:将连接到 WAL 文件的数据库名称。如果为NULL,则连接到所有数据库。如果不是NULL,则只连接到指定数据库的 WAL 文件。
- eMode:当此参数设置为 SQLITE_CHECKPOINT_PASSIVE 时,在 WAL 文件中找到所有活动的奇偶交错帧并将它们排序交错以便将单个帧作为 SQLite 数据库中的单个页面写入。否则,如果设置为 SQLITE_CHECKPOINT_RESTART 或 SQLITE_CHECKPOINT_TRUNCATE,则在 WAL 文件中找到所有活动的奇偶交错帧并将它们写入 SQLite 数据库中。如果它设置为 SQLITE_CHECKPOINT_FULL,则查找所有WAL文件,并找出其中活动的奇偶交错帧并将其写入SQLite数据库。
- pnLog:指向一个int的指针,如果不想接收WAL文件的大小,则传递NULL。
- pnCkpt:指向一个int的指针,如果不需要接收检查点的总页数,则传递NULL。

sqlite3_wal_checkpoint函数原型如下:

int sqlite3_wal_checkpoint(
  sqlite3 *db,
  const char *zDb
);

参数说明:

- db:由sqlite3_open或sqlite3_open_v2函数打开的一个数据库连接实例。
- zDb:将连接到 WAL 文件的数据库名称。如果为NULL,则连接到所有数据库。如果不是NULL,则只连接到指定数据库的 WAL 文件。

示例程序演示了如何使用sqlite3_wal_checkpoint_v2函数手动触发WAL检查点。

#include <sqlite3.h>
#include <stdio.h>
int main() {
    sqlite3 *db;
    char *err_msg = 0;
    sqlite3_open(":memory:", &db);
    sqlite3_exec(db, "PRAGMA journal_mode=WAL;", NULL, NULL, &err_msg);
    sqlite3_exec(db, "CREATE TABLE test (a, b, PRIMARY KEY(a, b));", NULL, NULL, &err_msg);
    sqlite3_exec(db, "BEGIN;", NULL, NULL, &err_msg);
    for (int i = 0; i < 10; i++) {
        char sql_insert[100];
        sprintf(sql_insert, "INSERT INTO test VALUES (%d, %d);", i, i);
        sqlite3_exec(db, sql_insert, NULL, NULL, &err_msg);
    }
    printf("WAL file size before: %d bytes\n", sqlite3_wal_checkpoint_v2(db, NULL, SQLITE_CHECKPOINT_FULL, NULL, NULL));
    sqlite3_exec(db, "COMMIT;", NULL, NULL, &err_msg);
    printf("WAL file size after: %d bytes\n", sqlite3_wal_checkpoint_v2(db, NULL, SQLITE_CHECKPOINT_FULL, NULL, NULL));
    sqlite3_close(db);
    return 0;
}

以上程序首先使用“PRAGMA journal_mode=WAL;”指令开启WAL模式,然后创建一个表“test”,并在其中添加一些数据。然后,通过sqlite3_wal_checkpoint_v2函数使用SQLITE_CHECKPOINT_FULL标志触发了一个手动检查点,并记录WAL日志文件的大小。最后提交事务并再次记录WAL日志文件大小。

程序输出的结果为:

WAL file size before: 12288 bytes
WAL file size after: 4096 bytes

可以看到,手动调用日志文件检查点已将WAL文件的大小从12288字节缩小到4096字节。这是一种有效节省空间的方法。

59.sqlite3_wal_hook

函数功能:可以在WAL模式下注册一个回调函数。当需要在WAL文件中写入数据时,该回调函数将被调用。

在WAL模式下,数据更改是在WAL文件中记录的。sqlite3_wal_hook回调函数允许应用程序检测正在写入到WAL日志文件的数据,可以自定义要写入到WAL日志文件的内容。这可以用于记录应用程序相关的元数据或其他一些额外的信息。

sqlite3_wal_hook函数原型如下:

void (*sqlite3_wal_hook)(
  sqlite3*, 
  int(*)(void *,sqlite3*,const char*,int),
  void*
);

参数说明:

- db:一个打开的数据库连接。
- xCallback:一个回调函数,由SQLite3调用以在WAL文件中写入数据。
- pArg:回调函数的第3个参数是指向此指针的。它是对回调函数的额外自定义数据,当回调函数被调用时,它作为第3个参数(完全解释由回调函数协议确定的额外数据)的值传递给回调函数。

其中,xCallback参数为回调函数的指针。

回调函数原型如下:

int (*xCallback)(void *pArg, sqlite3 *db, const char *zWal, int n)

参数说明:

- pArg:由sqlite3_wal_hook函数传递的第3个参数。
- db:数据库连接句柄。
- zWal:指向写入的数据的指针。
- n:写入的数据的长度。

该回调函数可以返回零或一个非零值。如果它返回非零,则将导致WAL日志文件上的写入被撤消,不会写入磁盘。

以下示例程序演示了在WAL文件中注册回调函数sqlite3_wal_hook。

#include <sqlite3.h>
#include <string.h>
#include <stdio.h>
static int wal_write_callback(void *pArg, sqlite3 *db, const char *zWal, int n) {
    printf("WAL write callback: %s (size=%d)\n", zWal, n);
    // 将二进制数据解析为int类型并输出
    if (n == sizeof(int)) {
        int data = 0;
        memcpy(&data, zWal, sizeof(int));
        printf("Data: %d\n", data);
    }
    return 0;
}
int main() {
    sqlite3 *db;
    sqlite3_open(":memory:", &db);
    sqlite3_wal_hook(db, wal_write_callback, NULL);
    // 将数据写入WAL文件中
    int data = rand();
    printf("Write data to WAL file: %d\n", data);
    sqlite3_wal_hook(db, NULL, NULL);  // 禁用回调以避免死循环
    sqlite3_wal_write(db, &data, sizeof(int));
    sqlite3_wal_hook(db, wal_write_callback, NULL);
    sqlite3_close(db);
    return 0;
}

以上程序打开一个in-memory数据库,使用sqlite3_wal_hook函数注册回调函数“wal_write_callback”,然后随机生成一个整数并使用sqlite3_wal_write函数将该整数写入WAL文件中。使用memcpy函数将二进制数据解析为int类型并输出。

运行该程序,输出如下:

Write data to WAL file: 1710216883
WAL write callback: �4� (size=4)
Data: 1710216883
以上程序通过调用sqlite3_wal_hook函数为WAL文件注册回调函数“wal_write_callback”,该函数在数据写入到WAL文件时会被调用。该函数被调用时,输出写入WAL文件的数据。

注意:回调函数中执行的操作不应太耗时,否则可能会影响系统的性能。

至此sqlite3类的所有函数介绍完成了,本类是sqlite数据库的核心,也是最重要的函数后续将陆续介绍其他的类与成员函数相关内容。敬请期待《我很快会回来的》。

后记:

本系列文章工作量太大,本人水平严重不足文章中错漏再所难免,也希望看到错误的朋友给与及时指正(在此先行谢过)。赶稿时间很冲忙也很累,希望大佬们不要太叫真,本文献给与我相同碰到一些问题无处找解答的人,若能帮助到您一点点,吾心深感荣幸!愿我们的技术一起进步!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/467963.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

统计-R(相关系数)与R^2(决定系数)

1.相关系数&#xff08;R&#xff09; 定义&#xff1a;考察两个事物&#xff08;在数据里我们称之为变量&#xff09;之间的相关程度。 假设有两个变量X&#xff0c;Y&#xff0c;那么两个变量间的皮尔逊相关系数可通过以下公式计算&#xff1a; 公式一&#xff1a; 其中…

OkHttp

文章目录 OkHttp概要1.简介2.特点3.基本组成5.工作流程 拦截器1.简介2.内置拦截器3.自定义拦截器 连接池1.简介2.常用参数配置选项 Dispatcher和线程池1.简介2.重要方法3.DispatCher中的双端队列4.总结 OkHttp 概要 1.简介 OkHttp是一个开源的HTTP客户端&#xff0c;用于在J…

【Sass】1px分割线 + 缩进分割线

效果图 1. 亮色模式效果 2. 暗色模式效果 设计思路 配色使用grey色 优点&#xff1a;无论在暗色模式还是亮色模式都可以看清楚分割线 使用after,before 伪元素绘制线条&#xff0c;并压缩线条transform: scaleY(.25) 注意事项 必须确保父级有宽高父级定位必须为position: r…

uniapp+vue3+setup语法糖开发微信小程序时不能定义globalData的解决方法

在使用 uniapp 开发小程序的时候&#xff0c; 发现使用了setup 语法糖 &#xff0c;定义 globalData 时&#xff0c;要不是定义不了&#xff0c; 要不就是使用 getApp()取不到&#xff0c;后来想到一个不伦不类的方法解决了&#xff0c; 这个方法有点难看&#xff0c; 但是解决…

文件读取的高效方法与设计模式

⭐️ 导言 在软件开发中&#xff0c;经常需要处理各种类型的文件&#xff0c;包括文本文件&#xff08;如JSON、CSV、TXT&#xff09;、Excel 文件等。针对不同类型的文件&#xff0c;我们需要选择合适的方法来读取和处理文件内容。在本篇博客中&#xff0c;小编以python为例&…

【Windows 常用工具系列 15 -- VMWARE ubuntu 安装教程】

文章目录 安装教程镜像下载 工具安装 安装教程 安装教程参考链接&#xff1a;https://blog.csdn.net/Python_0011/article/details/131619864 https://linux.cn/article-15472-1.html 激活码 VMware 激活码连接&#xff1a;https://www.haozhuangji.com/xtjc/180037874.html…

工控MCGS触摸屏Hacking勒索部署

https://github.com/MartinxMax/Mo0n_V1.2 !!不关注点赞收藏,以后没好东西了奥!! 端口扫描-获取信息 $python Mo0n.py -scan x.x.x.0/24 or $nmap -sS -Pn -T4 x.x.x.0/24 -p 127 MCGS编程软件 发现触摸屏受到密码保护 Oops&#xff01;&#xff01;&#xff01; echo /\_…

基于深度学习YOLOv8+Pyqt5的工地安全帽头盔佩戴检测识别系统(源码+跑通说明文件)

wx供重浩&#xff1a;创享日记 对话框发送&#xff1a;318安全帽 获取完整源码源文件7000张已标注的数据集训练好的模型配置说明文件 可有偿59yuan一对一远程操作配置环境跑通程序 效果展示&#xff08;图片检测批量检测视频检测摄像头检测&#xff09; 基于深度学习YOLOv8Pyqt…

【Excel自动化办公】使用openpyxl对Excel进行读写操作

目录 一、环境安装 1.1 创建python项目 1.2 安装openpyxl依赖 二、Excel数据读取操作 三、Excel数据写入操作 3.1 创建空白工作簿 3.2 写数据 四、设置单元格样式 4.1 字体样式 4.2 设置单元格背景填充色 4.3 设置单元格边框样式 4.4 单元格对齐方式 4.5 数据筛选…

B008-springcloud alibaba 短信服务 sms

目录 短信服务介绍短信服务使用准备工作阿里云官网实名认证开通短信服务申请认证秘钥申请短信签名申请短信模板 短信服务API介绍短信发送(SendSms)短信查询(QuerySendDetails)功能测试 下单之后发送短信 短信服务介绍 短信服务&#xff08;Short Message Service&#xff09;是…

08-热点文章-定时计算-黑马头条

xxl-Job分布式任务调度 1 今日内容 1.1 需求分析 目前实现的思路&#xff1a;从数据库直接按照发布时间倒序查询 问题1&#xff1a; 如何访问量较大&#xff0c;直接查询数据库&#xff0c;压力较大 问题2&#xff1a; 新发布的文章会展示在前面&#xff0c;并不是热点文章 …

SpringCloud搭建微服务之Micrometer分布式链路追踪

1. 概述 由于Spring Cloud Sleuth最新版本只支持Spring Boot 2.7.x&#xff0c;核心项目已经迁移到Micrometer Traceing项目&#xff0c;Spring Boot 3.x版本要实现分布式链路追踪需要集成Micrometer。更多详情可以参阅Micrometer官网 本文将以Spring Boot 3.2.x和Spring Clo…

Cinema 4D 2024 for mac/Win:开启三维动画与建模新纪元

在数字化时代&#xff0c;三维动画与建模已成为影视、游戏、广告等多个领域不可或缺的创作工具。而Cinema 4D&#xff0c;作为这一领域的佼佼者&#xff0c;始终以其卓越的性能和创新的功能引领着行业的发展。如今&#xff0c;Cinema 4D 2024的发布&#xff0c;更是为我们带来了…

【command not found】原因分析及解决

在使用Linux时&#xff0c;会经常遇到 “command not found” 的错误。错误信息提示的是&#xff1a;Linux没有找到该命令。原因主要分类有&#xff1a; 1.命令拼写错误 2.软件路径配置错误 3.Linux 系统就没有安装该命令。 一、确认命令没有拼写错误 Linux 中的所有命令都是…

使用CSS的object-position实现图片在img标签中的定位

在CSS中&#xff0c;object-position属性它允许我们精确地控制替换元素&#xff08;如<img>、<video>等&#xff09;内容在其容器内的位置。通过指定水平和垂直方向的偏移量&#xff0c;可以轻松地调整元素内容在容器内的起始点&#xff0c;实现精准定位。 1 语法…

Poly Kernel Inception Network在遥感检测中的应用

摘要 https://export.arxiv.org/pdf/2403.06258 遥感图像&#xff08;RSI&#xff09;中的目标检测经常面临一些日益严重的挑战&#xff0c;包括目标尺度的巨大变化和多样的上下文环境。先前的方法试图通过扩大骨干网络的空间感受野来解决这些挑战&#xff0c;要么通过大核卷积…

嵌入式-4种经典继电器驱动电路-单片机IO端口/三极管/达林顿管/嵌套连接

文章目录 一&#xff1a;继电器原理二&#xff1a;单片机驱动电路三&#xff1a;经典继电器驱动电路方案3.1 继电器驱动电路方案一&#xff1a;I/O端口灌电流方式的直接连接3.1.1 方案一的继电器特性要求3.1.2 方案一可能会损坏I/O口 3.2 继电器驱动电路方案二&#xff1a;三极…

深度学习-part3(反向传播、GAN)

5. 反向传播 5.1 什么是反向传播&#xff1f;‍ 通俗解释&#xff1a;类比几个人站成一排&#xff0c;第一个人看一幅画&#xff08;输入数据&#xff09;&#xff0c;描述给第二个人&#xff08;隐层&#xff09;……依此类推&#xff0c;到最后一个人&#xff08;输出&…

EDI在汽车主机厂配送流程中的应用

汽车主机厂的汽车配送流程始于汽车 “生产结束 ” &#xff0c;止于 “交付给经销商 ” 。在这个流程中&#xff0c;企业作为主机厂的下游供应商&#xff0c;与主机厂的物流服务供应商之间的信息交换将会变得十分重要。 配送流程&#xff1a;运输订单以及报告 汽车主机厂提供预…

Elasticsearch8.x版本Java客户端Elasticsearch Java API 如何并发修改

前言 并发控制&#xff0c;一般有两种方案&#xff0c;悲观锁和乐观锁&#xff0c;其中悲观锁是默认每次更新操作肯定会冲突&#xff0c;所以每次操作都要先获取锁&#xff0c;操作完毕再释放锁&#xff0c;适用于写比较多的场景。而乐观锁是默认每次更新操作都不会冲突&#…