SQLite3 C 教程

2019-12-20  本文已影响0人  wjundong

这是 SQLite 数据库的 C 编程教程,它介绍了使用 C 语言进行 SQLite 编程的基础知识。你可能还需要 SQLite tutorial 教程。

SQL数据库

SQLite3 工具

SQLite3 工具是 SQLite 库的基于终端的前端。它以交互方式来分析用户提供的查询 (queries) 操作,并以多种格式显示结果。它还可以在脚本中使用。它有自己的一组元命令,包括 .tables.load.database.dump。要获取所有指令的列表,我们只需键入. help 命令。

现在,我们将使用 sqlite3 工具创建一个新数据库。

$ sqlite3 test.db
SQLite version 3.22.0 2018-01-22 18:45:57
Enter ".help" for usage hints.

这里,我们为 sqlite3 工具提供了一个参数;test.db 是一个数据库名称。这是我们磁盘上的一个文件。如果它存在,则会将其打开。如果不是,则创建它。

sqlite> .tables
sqlite> .exit

tables 命令提供 test.db 数据库中所有存在的的表 (table)。当前没有表,因为我们没有创建它。exit 命令终止 sqlite3 命令行工具的交互会话。

退出后,通过 unix 的 ls 命令显示当前工作目录的内容。我们可以看到 test.db 文件。数据库会把所有数据都将存储在此单个文件中。

$ ls
test.db

C99

本教程使用 C99。对于 GNU C 编译器,我们需要使用 -std=c99 选项。对于 Windows 用户,推荐使用 PELES C IDE。(MSVC 不支持 C99。)。

int rc=sqlite3_open(“test.db”,&db);

在 C99 中,我们可以将声明与代码混合使用。但在旧的 C 程序中,我们需要将此行分成两行。

获取 SQLite 版本

在第一个代码示例中,我们将获得 SQLite 数据库的版本。

// -- 获取 SQLite 版本 v1 --
#include <sqlite3.h>
#include <stdio.h>

int main(void) 
{
    printf("%s\n", sqlite3_libversion()); 
    
    return 0;
}

sqlite3_libversion 函数的作用是:返回一个指示 SQLite 库的字符串。

头文件 sqlite3.h 定义了 SQLite 库提供给客户端程序的接口。它包含定义、函数原型和注释。它是 SQLite API 的权威来源。

我们使用 GNU C 编译器编译程序。然后运行它

$ gcc -o version version.c -lsqlite3 -std=c99
$ ./version
3.22.0

在第二个示例中,我们再次获取 SQLite 数据库的版本。但这次我们将使用 SQL 的查询 (query) 功能。

// -- 获取 SQLite 版本 v2 --
#include <sqlite3.h>
#include <stdio.h>

int main(void) 
{
    sqlite3 *db;
    sqlite3_stmt *pStmt;

    // 打开数据库
    if (sqlite3_open(":memory:", &db) != SQLITE_OK) {
        fprintf(stderr, "Cannot open database: %s\n", sqlite3_errmsg(db));
        sqlite3_close(db);
        return 1;
    }

    // 编译 SQL 语句
    if (sqlite3_prepare_v2(db, "SELECT SQLITE_VERSION()", -1, &pStmt, 0) != SQLITE_OK) {
        fprintf(stderr, "Failed to fetch data: %s\n", sqlite3_errmsg(db));
        sqlite3_close(db);
        return 1;
    }

    // 执行 SQL 语句
    if (sqlite3_step(pStmt) == SQLITE_ROW) 
        printf("%s\n", sqlite3_column_text(pStmt, 0));
    
    // 释放 SQL 语句句柄并关闭数据库
    sqlite3_finalize(pStmt);
    sqlite3_close(db);
    
    return 0;
}

SQLITE_VERSION() 查询用于获取 SQLite 库的版本。

sqlite3 *db;
sqlite3_stmt *pStmt;

sqlite3 结构定义了一个数据库句柄。每个打开的 SQLite 数据库由一个数据库句柄表示。
sqlite3_stmt 结构表示单个 SQL 语句 (statement)

sqlite3_open(":memory:", &db);

sqlite3_open 函数的作用是:打开一个数据库。它的参数是数据库名称和数据库句柄。
:memory: 是一个特殊的数据库名称,使用它可以新开一个数据库且该数据存放在内存中。函数的返回代码指示数据库是否已成功打开。成功打开并建立连接时返回 SQLITE_OK。

如果返回代码指示错误,这里我们会将消息打印到控制台,关闭数据库句柄,然后终止程序。
sqlite3_errmsg 函数的作用是:返回错误的描述。
无论打开数据库连接句柄时是否发生错误,都应该通过将其传递给 sqlite3_close() 函数来释放与数据库连接句柄关联的资源。

sqlite3_prepare_v2(db, "SELECT SQLITE_VERSION()", -1, &pStmt, 0);

就像 C/C++ 或者 Java 那样,需要先编译才能执行命令。所以在执行 SQL 语句之前,必须首先使用 sqlite3_prepare 系列函数之一将其编译成字节码 (sqlite3_prepare() 函数已弃用)。

sqlite3_prepare_v2() 函数接受五个参数:

如果成功,sqlite3_prepare_v2() 将返回 SQLITE_OK;否则将返回错误代码。

sqlite3_step(pStmt);

sqlite3_step() 函数用来运行 SQL 语句。SQLITE_ROW 返回代码表示已经准备好下一行 (row) 数据,这说明我们的代码已经被正确执行。因为我们的 SQL 语句就为了获取版本号,它只返回一条数据,因此,我们只调用此函数一次。如果一次执行中可能会返回多条数据,我们可以进行多次调用,每调用一次就获取一条数据,SQL 内部会自动向下迭代。

sqlite3_column_text(pStmt, 0);

正确执行后 pStmt 句柄包含当前所迭代到的数据的索引信息,我们可以用它来访问该条数据信息。

sqlite3_finalize(res) 函数的作用是:销毁我们准备好的语句对象。
最后sqlite3_close() 函数关闭了与数据库的连接。

插入数据

我们创建一个 CARS 表,并在其中插入几行。

// -- SQLite3 数据库插入数据的例子 --

#include <sqlite3.h>
#include <stdio.h>

int main(void) 
{
    
    sqlite3 *db;
    char *err_msg = 0;
    
    // 打开数据库
    if (sqlite3_open("test.db", &db) != SQLITE_OK) {
        fprintf(stderr, "Cannot open database: %s\n", sqlite3_errmsg(db));
        sqlite3_close(db);
        return 1;
    }
    
    // 设置 SQL 语句字符串
    char *sql = "DROP TABLE IF EXISTS Cars;" 
                "CREATE TABLE Cars(Id INT, Name TEXT, Price INT);" 
                "INSERT INTO Cars VALUES(1, 'Audi', 52642);" 
                "INSERT INTO Cars VALUES(2, 'Mercedes', 57127);" 
                "INSERT INTO Cars VALUES(3, 'Skoda', 9000);" 
                "INSERT INTO Cars VALUES(4, 'Volvo', 29000);" 
                "INSERT INTO Cars VALUES(5, 'Bentley', 350000);" 
                "INSERT INTO Cars VALUES(6, 'Citroen', 21000);" 
                "INSERT INTO Cars VALUES(7, 'Hummer', 41400);" 
                "INSERT INTO Cars VALUES(8, 'Volkswagen', 21600);";

    // 编译并执行 SQL 语句字符串
    if (sqlite3_exec(db, sql, 0, 0, &err_msg) != SQLITE_OK ) {
        fprintf(stderr, "SQL error: %s\n", err_msg);
        sqlite3_free(err_msg);        
        sqlite3_close(db);
        return 1;
    }
    
    sqlite3_close(db);

    return 0;
}

上面代码中,我们连接到 test.db 数据库,创建一个 CARS 表,并在创建的表中插入 8 行数据。

char *err_msg = 0;

如果出现错误,此指针将指向的错误消息。

然后是设置 SQL 语句字符串, 这些 SQL 语句会创建一个 CARS 表,并用数据填充它。语句必须用分号 ';'分隔。

rc = sqlite3_exec(db, sql, 0, 0, &err_msg);

sqlite3_exec 函数是 sqlite3_prepare_v2sqlite3_stepsqlite3_finalize 的便捷封装方式,它允许应用程序运行 SQL 的多条语句,而不必使用大量 C 代码。

sqlite3_free(err_msg);

必须使用 sqlite3_free() 函数调用释放分配的消息字符串。

我们可以使用 sqlite3 工具验证写入的数据。首先,我们修改数据在控制台中的显示方式。这里我们使用清爽的列对齐显示模式,并要求显示每条数据的标题。然后执行 SELECT * FROM Cars;语句。

sqlite> .mode column  
sqlite> .headers on
sqlite> SELECT * FROM Cars;
Id          Name        Price     
----------  ----------  ----------
1           Audi        52642     
2           Mercedes    57127     
3           Skoda       9000      
4           Volvo       29000     
5           Bentley     350000    
6           Citroen     21000     
7           Hummer      41400     
8           Volkswagen  21600

这就是我们写入CARS表的数据。

最后插入的行ID

有时,我们需要确定最后插入的行的 ID。为此,我们使用 sqlite3_last_insert_rowid 函数。如下

int last_id = sqlite3_last_insert_rowid(db);
printf("The last Id of the inserted row is %d\n", last_id);

下面语句执行后会在内存中创建一个 Friends 表。其 ID 列自动递增。

char *sql = "CREATE TABLE Friends(Id INTEGER PRIMARY KEY, Name TEXT);"
            "INSERT INTO Friends(Name) VALUES ('Tom');"
            "INSERT INTO Friends(Name) VALUES ('Rebecca');"
            "INSERT INTO Friends(Name) VALUES ('Jim');"
            "INSERT INTO Friends(Name) VALUES ('Roger');"
            "INSERT INTO Friends(Name) VALUES ('Robert');";

在 SQLite 中,整数主键 (PRIMARY KEY) 列会自动递增。还有一个 AUTOINCREMENT 关键字。当该关键字应用于整数主键时,使用的 ID 创建算法略有不同。当使用自动递增列时,我们需要显式地声明列名。

检索数据

我们已经将一些数据插入到 test.db 数据库中。在下面的示例中,我们将从数据库检索数据。

// -- 检索数据 Callback方法

#include <sqlite3.h>
#include <stdio.h>

int callback(void *, int, char **, char **);

int main(void) {
    
    sqlite3 *db;
    char *err_msg = 0;

    // 打开数据库
    if (sqlite3_open("test.db", &db) != SQLITE_OK) {
        fprintf(stderr, "Cannot open database: %s\n", 
                sqlite3_errmsg(db));
        sqlite3_close(db);
        return 1;
    }
    
    // 设置语句并编译执行
    char *sql = "SELECT * FROM Cars;";
    if (sqlite3_exec(db, sql, callback, 0, &err_msg) != SQLITE_OK ) {    
        fprintf(stderr, "Failed to select data\n");
        fprintf(stderr, "SQL error: %s\n", err_msg);
        sqlite3_free(err_msg);
        sqlite3_close(db);
        return 1;
    } 

    sqlite3_close(db);
    
    return 0;
}

// 回调函数
int callback(void *NotUsed, int argc, char **argv, char **azColName) {
    NotUsed = 0;
    for (int i = 0; i < argc; i++)
        printf("%s = %s\n", azColName[i], argv[i] ? argv[i] : "NULL");
    printf("\n");
    
    return 0;
}

我们使用 SELECT * FROM Cars; 语句从 CARS 表中检索所有行。

int callback(void *, int, char **, char **);

这是与 sqlite3_exec() 函数结合使用的回调函数的函数原型。

sqlite3_exec(db, sql, callback, 0, &err_msg);

这次 sqlite3_exec 函数中的我们加入了回调函数,sqlite3_exec 在处理求值语句时每得到一个结果行将调用一次 callback 函数。

int callback(void *NotUsed, int argc, char **argv, char **azColName) {
    NotUsed = 0;
    for (int i = 0; i < argc; i++)
        printf("%s = %s\n", azColName[i], argv[i] ? argv[i] : "NULL");
    printf("\n");
    
    return 0;
}

来看这个回调函数

所以上面代码中,我们遍历结果行标题行中的所有列就并打印出了它们的名称和内容。

$ ./select_all 
Id = 1
Name = Audi
Price = 52642

Id = 2
Name = Mercedes
Price = 57127

...

以上是示例的部分输出。

参数化查询

现在我们将用到参数化查询。参数化查询,也称为预准备语句 (prepared statements), 可提高安全性和性能。当我们使用参数化查询时,我们使用占位符,而不是直接将值写入语句。

// -- SQLite3 参数化查询例子 --

#include <sqlite3.h>
#include <stdio.h>

int main(void) {
    
    sqlite3 *db;
    char *err_msg = 0;
    sqlite3_stmt *pStmt;
    
    // 打开数据库
    if (sqlite3_open("test.db", &db) != SQLITE_OK) {
        fprintf(stderr, "Cannot open database: %s\n", sqlite3_errmsg(db));
        sqlite3_close(db);
        return 1;
    }
    
    // 设置并预编译 SQL 语句
    char *sql = "SELECT Id, Name FROM Cars WHERE Id = ?";
    if (sqlite3_prepare_v2(db, sql, -1, &pStmt, 0) == SQLITE_OK)
        // 如果预编译成功,则对占位符进行绑定:第一个占位符,设置为 3
        sqlite3_bind_int(pStmt, 1, 3);
    else 
        fprintf(stderr, "Failed to execute statement: %s\n", sqlite3_errmsg(db));

    // 执行语句获取结果行,如果数据可用则获取数据
    if (sqlite3_step(pStmt) == SQLITE_ROW) {
        printf("%s: ", sqlite3_column_text(pStmt, 0));
        printf("%s\n", sqlite3_column_text(pStmt, 1));
    } 

    sqlite3_finalize(pStmt);
    sqlite3_close(db);
    
    return 0;
}

在本例中,问号 '?' 用作占位符,稍后将其替换为实际值并为 SQL 查询提供 Id。

char *sql = "SELECT Id, Name FROM Cars WHERE Id = ?";
sqlite3_bind_int(pStmt, 1, 3);

sqlite3_bind_int 将整数值绑定到准备好的语句,占位符替换为整数值3。

sqlite3_step 函数将执行处理好的 SQL 语句,并索引到执行结果行。

如果有某行数据可用,我们将使用 sqlite3_column_text 函数获取结果行中的两列的值,即

printf("%s: ", sqlite3_column_text(res, 0));
printf("%s\n", sqlite3_column_text(res, 1));
$ ./parameterized 
3: Skoda

正如所期待的,该示例返回第三个数据的 Id 和汽车的名称,因为我们绑定到占位符的数据是 3 。

第二个示例使用带有命名占位符的参数化语句

// -- SQLite3 使用命名占位例子 --

#include <sqlite3.h>
#include <stdio.h>

int main(void) 
{
    
    sqlite3 *db;
    char *err_msg = 0;
    sqlite3_stmt *pStmt;

    // 打开数据库
    if (sqlite3_open("test.db", &db) != SQLITE_OK) {
        fprintf(stderr, "Cannot open database: %s\n", sqlite3_errmsg(db));
        sqlite3_close(db);
        return 1;
    }
    
    // 设置并预编译 SQL 语句
    char *sql = "SELECT Id, Name FROM Cars WHERE Id = @id";
    if (sqlite3_prepare_v2(db, sql, -1, &pStmt, 0) != SQLITE_OK) {
        fprintf(stderr, "Failed to execute statement: %s\n", sqlite3_errmsg(db));
        sqlite3_close(db);
        return 1;
    }

    // 通过占位符名 @name 获取占位符位置
    int idx = sqlite3_bind_parameter_index(pStmt, "@id");
    int value = 4;

    // 把整数值绑定到语句中的占位符位置
    sqlite3_bind_int(pStmt, idx, value);
    
    // 执行语句获取结果行,如果数据可用则打印数据
    if (sqlite3_step(pStmt) == SQLITE_ROW) {
        printf("%s: ", sqlite3_column_text(pStmt, 0));
        printf("%s\n", sqlite3_column_text(pStmt, 1));
    } 

    sqlite3_finalize(pStmt);
    sqlite3_close(db);
    
    return 0;
}

命名占位符的前缀可以是 '@', 也可以是冒号 ':'

int idx = sqlite3_bind_parameter_index(pStmt, "@id");

sqlite3_bind_parameter_index 函数的作用是返回给定占位符名称的 SQL 占位符的索引。

插入图片

在本节中,我们将向 SQLite 数据库插入图像。请注意,有些人反对将图像放入数据库。这里我们只演示如何做到这一点。我们没有详细讨论是否应该将图像保存在数据库中的技术问题。

sqlite> CREATE TABLE Images(Id INTEGER PRIMARY KEY, Data BLOB);

对于本例,我们创建一个名为 Images 的新表。其中我们使用 BLOB 数据类型来存放图片,它代表二进制大对象。

#include <sqlite3.h>
#include <stdio.h>
#include <stdlib.h>

int main(int argc, char **argv) {

    FILE *fp = fopen("woman.jpg", "rb");
    
    if (fp == NULL) {
        fprintf(stderr, "Cannot open image file\n");    
        return 1;
    }

    // 获取文件大小,在这里为了简洁,我们没有对获取过程
    // 可能出现的出错进行处理,实际开发应注意这一点
    fseek(fp, 0, SEEK_END);
    int fsize = ftell(fp);
    fseek(fp, 0, SEEK_SET);

    // 根据获取文件大小获取内存,并将文件读入到内存
    char *data = (char*)malloc(sizeof(char)*fsize);
    if(data == NULL || fread(data, 1, fsize, fp) != fsize) {
        fprintf(stderr, "malloc or readfile error\n");
        fclose(fp);
        return 1;
    }
    fclose(fp);

    sqlite3 *db;
    char *err_msg = 0;
    sqlite3_stmt *pStmt;

    // 打开数据库
    if (sqlite3_open("test.db", &db) != SQLITE_OK) {
        fprintf(stderr, "Cannot open database: %s\n", sqlite3_errmsg(db));
        sqlite3_close(db);
        return 1;
    }
    
    // 语句预处理
    char *sql = "INSERT INTO Images(Data) VALUES(?)";    
    if (sqlite3_prepare(db, sql, -1, &pStmt, 0) != SQLITE_OK) {
        fprintf(stderr, "Cannot prepare statement: %s\n", sqlite3_errmsg(db));
        sqlite3_close(db);
        return 1;
    }    
    
    // 绑定占位符并插入内存中的二进制文件
    sqlite3_bind_blob(pStmt, 1, data, fsize, SQLITE_STATIC);
    if (sqlite3_step(pStmt) != SQLITE_DONE)
        printf("execution failed: %s", sqlite3_errmsg(db));
    
    sqlite3_finalize(pStmt);    
    sqlite3_close(db);

    return 0;
}

在此程序中,我们从当前工作目录读取图像到堆内存中,之后将其写入 SQLite test.db 数据库的 Images 表。

sqlite3_bind_blob(pStmt, 1, data, size, SQLITE_STATIC);

sqlite3_bind_blob 函数的作用是:将二进制数据绑定到编译后的语句中的占位符。SQLITE_STATIC 参数表示指向内容信息的指针是静态的,不需要由 SQLite 释放,这将由我们主动释放。

sqlite3_step(pStmt);

最后调用 sqlite3_step 执行上面处理好的 SQLite 语句 pStmt 便可以将图形写入到表中。

读取图像

在本节中,我们将执行相反的操作,从数据库表中读取图像。

#include <sqlite3.h>
#include <stdio.h>


int main(void) {
    
    FILE *fp = fopen("2233.jpg", "wb");
    if (fp == NULL) {
        fprintf(stderr, "Cannot open image file\n");    
        return 1;
    }    
    
    sqlite3 *db;
    char *err_msg = 0;
    sqlite3_stmt *pStmt;

    // 打开数据库
    if (sqlite3_open("test.db", &db) != SQLITE_OK) {
        fprintf(stderr, "Cannot open database: %s\n", sqlite3_errmsg(db));
        sqlite3_close(db);
        return 1;
    }
    
    // 语句预处理
    char *sql = "SELECT Data FROM Images WHERE Id = 1";
    if (sqlite3_prepare_v2(db, sql, -1, &pStmt, 0) != SQLITE_OK) {
        fprintf(stderr, "Cannot prepare statement: %s\n", sqlite3_errmsg(db));
        sqlite3_close(db);
        return 1;
    }
    
    // 执行语句
    if (sqlite3_step(pStmt) == SQLITE_ROW) {
        // 先获取对应列数据的大小
        int bytes = sqlite3_column_bytes(pStmt, 0);
        // 获取二进制文件并将其写入文件
        fwrite(sqlite3_column_blob(pStmt, 0), bytes, 1, fp);
    }

    fclose(fp);
    sqlite3_finalize(pStmt);   
    sqlite3_close(db);
    
    return 0;
}
bytes = sqlite3_column_bytes(pStmt, 0);
fwrite(sqlite3_column_blob(pStmt, 0), bytes, 1, fp);

sqlite3_column_bytes 函数的作用是:返回 BLOB 中的字节数。
sqlite3_column_blob 函数返回指向所选二进制数据的指针。

元数据

元数据是关于数据库中数据的信息。SQLite 中的元数据包含有关表和列的信息,我们在这些表和列中存储数据。受 SQL 语句影响的行数是元数据。结果集中返回的行数和列数也属于元数据。

可以使用 Pragma 命令获取 SQLite 中的元数据。SQLite 对象可能具有属性,这些属性是元数据。最后,我们还可以通过查询 SQLite 系统 sqlite_master 表来获得特定的元数据。

#include <sqlite3.h>
#include <stdio.h>

int callback(void *, int, char **, char **);

int main(void) {
    
    sqlite3 *db;
    char *err_msg = 0;
    
    // 打开数据库
    if (sqlite3_open("test.db", &db) != SQLITE_OK) {
        fprintf(stderr, "Cannot open database: %s\n", sqlite3_errmsg(db));
        sqlite3_close(db);
        return 1;
    }

    // 编译并执行语句
    char *sql = "PRAGMA table_info(Cars);";
    if (sqlite3_exec(db, sql, callback, 0, &err_msg) != SQLITE_OK ) {
        fprintf(stderr, "Failed to select data %s\n", err_msg);
        sqlite3_free(err_msg);
        sqlite3_close(db);
        return 1;
    }
    
    sqlite3_close(db);
    
    return 0;
}

int callback(void *NotUsed, int argc, char **argv, char **azColName) 
{    
    NotUsed = 0;
    for (int i = 0; i < argc; i++)
        printf("%s = %s\n", azColName[i], argv[i] ? argv[i] : "NULL");
    printf("------------\n");
    return 0;
}

在本例中,我们执行语句 PRAGMA table_info(Cars);,以获取有关 CARS 表的一些元数据信息。结果集中的列包括列序号、列名、数据类型、列是否可以为NULL以及列的默认值。

$ ./column_names
cid = 0
name = Id
type = INT
notnull = 0
dflt_value = NULL
pk = 1
------------
cid = 1
name = Name
type = TEXT
notnull = 0
dflt_value = NULL
pk = 0
------------
cid = 2
name = Price
type = INT
notnull = 0
dflt_value = NULL
pk = 0
------------

在下一个与元数据相关的示例中,我们将列出test.db数据库中的所有表

#include <sqlite3.h>
#include <stdio.h>


int callback(void *, int, char **, char **);


int main(void) {
    
    sqlite3 *db;
    char *err_msg = 0;
    
    if (sqlite3_open("test.db", &db) != SQLITE_OK) {
        fprintf(stderr, "Cannot open database: %s\n", sqlite3_errmsg(db));
        sqlite3_close(db);
        return 1;
    }
    
    char *sql = "SELECT name FROM sqlite_master WHERE type='table'";
    if (sqlite3_exec(db, sql, callback, 0, &err_msg) != SQLITE_OK ) {
        fprintf(stderr, "SQL error: %s\n", err_msg);
        sqlite3_free(err_msg);
        sqlite3_close(db);
        return 1;
    } 
    
    sqlite3_close(db);
    
    return 0;
}

int callback(void *NotUsed, int argc, char **argv, char **azColName) {
    NotUsed = 0;
    for (int i = 0; i < argc; i++) 
        printf("%s\n", argv[i] ? argv[i] : "NULL");
    return 0;
}

代码示例将当前数据库中的所有可用表打印到终端。

$ ./list_tables 
Friends
Cars
Images

以上就是之前操作的数据内容。

事务

事务是针对一个或多个数据库中的数据执行的数据库操作的原子单元。事务中所有 SQL 语句的效果可以全部提交到数据库,也可以全部回滚。

在 SQLite 中,除 SELECT 之外的任何命令都将启动隐式事务。此外,在事务中,像 create table.、Vacuum、Pragma 这样的命令将在执行之前提交以前的更改。

手动事务以 BEGIN TRANSACTION 语句开始,以 COMMIT 或 ROLLBACK 语句结束。

SQLite 支持三个非标准事务级别:延迟、立即和独占。

自动提交

默认情况下,SQLite 版本 3 在自动提交模式下运行。在自动提交模式下,与当前数据库连接关联的所有操作完成后,立即提交对数据库的所有更改。自动提交模式由 BEGIN 语句禁用,由 COMMIT 或 ROLLBACK 重新启用。

#include <sqlite3.h>
#include <stdio.h>

int main() {
    
    sqlite3 *db;
    
    if (sqlite3_open("test.db", &db) != SQLITE_OK) {
        fprintf(stderr, "Cannot open database: %s\n", sqlite3_errmsg(db));
        sqlite3_close(db);
        return 1;
    }    
    
    printf("Autocommit: %d\n", sqlite3_get_autocommit(db));
    sqlite3_close(db);
    
    return 0;
}
printf("Autocommit: %d\n", sqlite3_get_autocommit(db));

此示例检查数据库是否处于自动提交模式,如果数据库未处于自动提交模式,则 sqlite3_get_autoCommit() 函数返回零。如果处于自动提交模式,则返回非零值。

$ ./get_ac_mode 
Autocommit: 1

该示例确认 SQLite 默认处于自动提交模式。

下一个示例进一步阐明了自动提交模式。在自动提交模式下,每个非 SELECT 语句都是一个立即提交的小事务。

#include <sqlite3.h>
#include <stdio.h>

int main(void) {
    
    sqlite3 *db;
    char *err_msg = 0;
    

    if (sqlite3_open("test.db", &db) != SQLITE_OK) {
        fprintf(stderr, "Cannot open database: %s\n", sqlite3_errmsg(db));
        sqlite3_close(db);
        return 1;
    }
    
    char *sql = "DROP TABLE IF EXISTS Friends;" 
                "CREATE TABLE Friends(Id INTEGER PRIMARY KEY, Name TEXT);" 
                "INSERT INTO Friends(Name) VALUES ('Tom');" 
                "INSERT INTO Friends(Name) VALUES ('Rebecca');" 
                "INSERT INTO Friends(Name) VALUES ('Jim');" 
                "INSERT INTO Friend(Name) VALUES ('Robert');";

    if (sqlite3_exec(db, sql, 0, 0, &err_msg) != SQLITE_OK ) {
        fprintf(stderr, "SQL error: %s\n", err_msg);
        sqlite3_free(err_msg);        
        sqlite3_close(db);
        return 1;
    } 
    
    sqlite3_close(db);
    
    return 0;
}

我们创建 Friends 表,并尝试用数据填充它,最后一条 SQL 语句有一个错误;因为没有 Friend 表。

$ ./autocommit 
SQL error: no such table: Friend
$ sqlite3 test.db
sqlite> .tables
Cars     Friends  Images 
sqlite> SELECT * FROM Friends;
1|Tom
2|Rebecca
3|Jim

因为每插入一个就自动提交一次,所以虽然插入最后一个数据的时候发生了错误,但是前面的三个数据已经提交到数据库里了。

交易

在下一个示例中,我们将一些 SQL 语句插入到事务中

#include <sqlite3.h>
#include <stdio.h>

int main(void) {
    
    sqlite3 *db;
    char *err_msg = 0;
    
    if (sqlite3_open("test.db", &db) != SQLITE_OK) {
        fprintf(stderr, "Cannot open database: %s\n", sqlite3_errmsg(db));
        sqlite3_close(db);
        return 1;
    }
    
    char *sql = "DROP TABLE IF EXISTS Friends;"
                "BEGIN TRANSACTION;" 
                "CREATE TABLE Friends(Id INTEGER PRIMARY KEY, Name TEXT);" 
                "INSERT INTO Friends(Name) VALUES ('Tom');" 
                "INSERT INTO Friends(Name) VALUES ('Rebecca');" 
                "INSERT INTO Friends(Name) VALUES ('Jim');" 
                "INSERT INTO Friend(Name) VALUES ('Robert');"
                "COMMIT;";
                           
    if (sqlite3_exec(db, sql, 0, 0, &err_msg) != SQLITE_OK ) {
        fprintf(stderr, "SQL error: %s\n", err_msg);
        sqlite3_free(err_msg);        
        sqlite3_close(db);
        return 1;
    }
        
    
    sqlite3_close(db);
    
    return 0;
}

第一个语句如果存在表 Friends 则将其删除。其他语句则放在事务中。事务在全有或全无模式下工作。要么什么都不做,要么什么都做。

sqlite> .tables
Cars    Images

因为最后一条语句有错误,所以事务被回滚,并且没有创建Friends表。

上一篇 下一篇

猜你喜欢

热点阅读