sqlite3数据库高效API(六)
sqlite3_open_v2sqlite3_prepare_v2 sqlite3_bind_xxx sqlite3_step sqlite3_finalize sqlite3_column_blob sqlite3_column_double sqlite3_column_value sqlite3_column_bytes等接口详解
sqlite3_open_v2
int sqlite3_open_v2(
const char *filename, /* Database filename (UTF-8) */
sqlite3 **ppDb, /* OUT: SQLite db handle */
int flags, /* Flags */
const char *zVfs /* Name of VFS module to use */
)
参数1:sqlite数据库文件名;
参数2:输出参数,数据库句柄;
参数3:标志,如下:
/* Ok for sqlite3_open_v2() 只读模式打开,若数据库不存在,则返回错误*/
#define SQLITE_OPEN_READONLY 0x00000001
/* Ok for sqlite3_open_v2() 若可能,将打开数据库进行读写,或仅在文件受操作系统写保护时才进行读取;这两种情况下数据库必须依据存在,否则返回错误;*/
#define SQLITE_OPEN_READWRITE 0x00000002
/* Ok for sqlite3_open_v2() 数据库不存则创建*/
#define SQLITE_OPEN_CREATE 0x00000004
#define SQLITE_OPEN_DELETEONCLOSE 0x00000008 /* VFS only */
#define SQLITE_OPEN_EXCLUSIVE 0x00000010 /* VFS only */
#define SQLITE_OPEN_AUTOPROXY 0x00000020 /* VFS only */
/* Ok for sqlite3_open_v2() 若设置了此标志,则文件名可解释成URL */
#define SQLITE_OPEN_URI 0x00000040
/* Ok for sqlite3_open_v2() 数据库将作为内存数据库打开;*/
#define SQLITE_OPEN_MEMORY 0x00000080
#define SQLITE_OPEN_MAIN_DB 0x00000100 /* VFS only */
#define SQLITE_OPEN_TEMP_DB 0x00000200 /* VFS only */
#define SQLITE_OPEN_TRANSIENT_DB 0x00000400 /* VFS only */
#define SQLITE_OPEN_MAIN_JOURNAL 0x00000800 /* VFS only */
#define SQLITE_OPEN_TEMP_JOURNAL 0x00001000 /* VFS only */
#define SQLITE_OPEN_SUBJOURNAL 0x00002000 /* VFS only */
#define SQLITE_OPEN_MASTER_JOURNAL 0x00004000 /* VFS only */
/* Ok for sqlite3_open_v2() 新的数据库连接将使用"多线程模式".意味着运行单独的线程同时使用SQLite,只要每个线程使用不同数据库连接;*/
#define SQLITE_OPEN_NOMUTEX 0x00008000
/* Ok for sqlite3_open_v2() 新的数据库将使用"序列化"线程模式;意味多个线程可以安全的尝试同时使用同一数据库连接;*/
#define SQLITE_OPEN_FULLMUTEX 0x00010000
/* Ok for sqlite3_open_v2() 数据库已启用共享缓存;*/
#define SQLITE_OPEN_SHAREDCACHE 0x00020000
/* Ok for sqlite3_open_v2() 数据库被打开共享缓存禁用;*/
#define SQLITE_OPEN_PRIVATECACHE 0x00040000
#define SQLITE_OPEN_WAL 0x00080000 /* VFS only */
参数4:sqlite3_vfs对象的名称,该对象定义了新数据库连接应该使用的操作系统接口; NULL表示使用默认的sqlite3_vfs对象;
sqlite3_prepare_v2
int sqlite3_prepare_v2(
sqlite3 *db, /* Database handle. */
const char *zSql, /* UTF-8 encoded SQL statement. */
int nBytes, /* Length of zSql in bytes. */
sqlite3_stmt **ppStmt, /* OUT: A pointer to the prepared statement */
const char **pzTail /* OUT: End of parsed string */
)
首选sqlite3_prepare_v2( ); sqlite3_prepare( )接口是遗留的,应该避免; sqlite3_prepare_v3( )有一个二外的"prepFlags"选项,用于特殊用途;
参数1:sqlite3_open_v2等获取的数据句柄;
参数2:utf8编码格式的sql语句;
参数3:nBytes<0,则zSql被读取到第一个零终止符; nBytes=0,不会生成任何准备好的语句;nBytes>0,表示从zSql语句读取的字节数;
参数4:ppStmt指向一个编译好的准备好的语句,可以使用sqlite3_step()执行; 若参数2输入的文本不包含SQL(输入的是空字符串或注释),则*ppStmt设置为NULL; 调用过程负责在编译完成后使用sqlite3_finalize()删除已编译的SQL语句;
返回值:指向成功返回SQLITE_OK;
sqlite3_bind_xxx
将值绑定到准备好的语句
int sqlite3_bind_blob(sqlite3_stmt*, int, const void*, int n, void(*)(void*));
int sqlite3_bind_double(sqlite3_stmt*, int, double);
int sqlite3_bind_int(sqlite3_stmt*, int, int);
int sqlite3_bind_int64(sqlite3_stmt*, int, sqlite3_int64);
int sqlite3_bind_null(sqlite3_stmt*, int);
int sqlite3_bind_text(sqlite3_stmt*,int,const char*,int,void(*)(void*));
int sqlite3_bind_value(sqlite3_stmt*, int, const sqlite3_value*);
int sqlite3_bind_pointer(sqlite3_stmt*, int, void*, const char*,void(*)(void*));
int sqlite3_bind_zeroblob(sqlite3_stmt*, int, int n);
参数1: 由sqlite3_prepare_v2()返回的sqlite3_stmt对象的指针;
参数2: 要设置的sql参数的索引;最左的sql参数的索引为1;当多次使用同一个命名的sql参数时,第二次和后续出现的索引与第一次出现的索引相同; 填1即可;
参数3: 要绑定到参数的值;若上述有该参数的是NULL指针,则忽略参数4;结果与sqlite3_bind_null相同;
参数4: 是参数3的字节数(字节而不是字符)
参数5:指示或控制参数3引用的对象的生存期; (1)可以传递一个析构函数,在sqlite处理完blob或字符串后处理它;即使对绑定API的调用失败,也会调用该函数来处理blob或字符串,但若参数3为NULL指针或参数4为负数,则不会调用析构函数; (2)可以传递常量SQLITE_STATIC,以知识应用程序任然负责处理对象;这种情况下,对象及其提供的指针必须保持有效,直到完成准备好的语句或将同一sql参数绑定到其他对象; (3)可以传递常量SQLITE_TRANSENT,以指示从sqlite3_bind_* 返回之前复制对象;
返回值: 成功返回SQLITE_OK;
sqlite3_step
int sqlite3_step(sqlite3_stmt*);
在使用了sqlite3_prepare_v2()等准备好语句后,必须调用一次或多次该函数来评估sqlite语句;
SQLITE_DONE表示语句已经成功执行完毕;
sqlite3_finalize
int sqlite3_finalize(sqlite3_stmt *pStmt);
调用sqlite3_finalize()函数以删除准备好的语句;
若最近换一次求值没有遇到错误,或者从未对语句求值 ,则sqlite3_finalize()返回SQLITE_OK;
6. 返回错误码
如果字符串或BLOB的大小超过sqlite3_limit(SQLITE_limit_LENGTH)或SQLITE_MAX_LENGTH所施加的限制,则可能返回SQLITE_TOOBIG。
如果参数索引超出范围,则返回SQLITE_RANGE。
如果malloc()失败,则返回SQLITE_NOMEM。
7. 查询相关API
const void *sqlite3_column_blob(sqlite3_stmt*, int iCol); double sqlite3_column_double(sqlite3_stmt*, int iCol); int sqlite3_column_int(sqlite3_stmt*, int iCol); const unsigned char *sqlite3_column_text(sqlite3_stmt*, int iCol); sqlite3_value *sqlite3_column_value(sqlite3_stmt*, int iCol); int sqlite3_column_bytes(sqlite3_stmt*, int iCol); int sqlite3_column_type(sqlite3_stmt*, int iCol);
上述例程返回有关查询的当前结果行的当个列的信息; 在每一种情况下,第一个参数都是一个sqlite3_stmt指针,由sqlite3_prepare_v2()获取; 第二个参数是返回其列的索引,结果最左侧的列的索引为0; 可以由sqlite3_column_count()确定结果中的列数;
若SQL语句当前未指向有效行,或者列索引超出范围,则结果未定义;只有在最近一次对sqlite3_step()的调用返回SQLITE_ROW并且既没有调用sqlite3_reset(),也没有调用sqlite3_finalize()时,才调用这些例程;
sqlite3_column_type()返回结果列的初始数据类型的数据类型代码;值是SQLITE_INTEGER、SQLITE_FLOAT、SQLITE_TEXT、SQLITE_BLOB或SQLITE_NULL之一;
若结果是BLOB或utf-8字符串,则sqlite3_column_bytes()接口来确定该BLOB或字符串的大小;返回的值是字符串的字节数而不是字符数;
8. 示例
#include <sqlite3.h>
#include <stdio.h>
#include <stdlib.h>
#include <time.h>
#include <unistd.h>
#include <string.h>
// 一条速度 信息
struct dr_speed_t
{
time_t start_time; // 时间
unsigned char speed; // 平均速度
unsigned char state; // 状态信息
unsigned char reserved[2];
};
static sqlite3 *g_db = NULL;
//typedef int (*sqlite3_callback)(void*,int,char**, char**);
static int count_callback(void *args, int nCol, char **rowArr, char **colName)
{
int *total_size = (int *)args;
*total_size = atoi(*rowArr);
return 0;
}
static int create_callback(void *args, int nCol, char **rowArr, char **colName)
{
sqlite3 *db = (sqlite3 *)args;
printf("nCol = %d %s \n",nCol, rowArr[0]);
if(atoi(rowArr[0]) < 1){
char sql[128], *err;
int ret;
snprintf(sql, sizeof(sql), "create table %s(tim INTEGER primary key, type INTEGER, bval blob)", "tab_product");
ret = sqlite3_exec(db, sql, 0, 0, &err);
if(ret != SQLITE_OK){
if(err){
sqlite3_free(err);
return -1;
}
}
}
return 0;
}
int SqlSave(time_t tim, void *extData, int dataLen)
{
char sql[128] = {0};
char *err = NULL;
int ret ;
sqlite3_stmt *stmt = NULL;
snprintf(sql, sizeof(sql),"replace into %s(tim,type,bval) values(%ld,%d,?)","tab_product",tim,0);
ret = sqlite3_prepare_v2(g_db, sql, -1, &stmt, NULL);
if(ret != SQLITE_OK){
printf("sqlite3_prepare_v2 err\n");
return -1;
}
ret = sqlite3_bind_blob(stmt, 1, extData, dataLen, NULL);
if(ret != SQLITE_OK){
goto ERR;
}
ret = sqlite3_step(stmt);
if(ret != SQLITE_OK){
goto ERR;
}
sqlite3_finalize(stmt);
return 0;
ERR:
sqlite3_finalize(stmt);
return -1;
}
int SqlQueryCount(time_t start_time, time_t end_time)
{
char sql[256], *err;
int ret;
int total_size = 0;
snprintf(sql, sizeof(sql), "select count(*) from %s where tim >= %ld and tim < %ld", "tab_product", start_time, end_time);
ret = sqlite3_exec(g_db, sql, count_callback, &total_size, &err);
if(ret != SQLITE_OK){
printf("sqlite3_exec err\n");
return -1;
}
return total_size;
}
int SqlQuerySpeed(time_t start_time, time_t end_time)
{
char sql[128] = {0};
char *err = NULL;
int ret ;
printf("query count = %d\n",SqlQueryCount(start_time, end_time));
sqlite3_stmt *stmt = NULL;
snprintf(sql, sizeof(sql),"select * from %s where tim > %ld and tim < %ld order by tim desc","tab_product",start_time, end_time);
ret = sqlite3_prepare_v2(g_db, sql, -1, &stmt, NULL);
if(ret != SQLITE_OK){
printf("sqlite3_prepare_v2 err\n");
return -1;
}
while(1)
{
ret = sqlite3_step(stmt);
if(ret != SQLITE_ROW){
printf("query exit\n");
break;
}
time_t tim = sqlite3_column_int(stmt, 0);
int type = sqlite3_column_int(stmt, 1);
struct dr_speed_t *speed = (struct dr_speed_t *)sqlite3_column_blob(stmt, 2);
int blob_len = sqlite3_column_bytes(stmt, 2);
if(blob_len != sizeof(struct dr_speed_t)){
printf("--> query err\n");
}
printf("speed=%d time=%ld state=%d\n",speed->speed,speed->start_time,speed->state);
}
sqlite3_finalize(stmt);
return 0;
}
int main(int argc, const char *argv[])
{
char sql[128] = {0};
char *err = NULL;
int ret ;
//读写,不存在则创建,多线程访问;
ret = sqlite3_open_v2("./config.db", &g_db, SQLITE_OPEN_READWRITE|SQLITE_OPEN_CREATE|SQLITE_OPEN_FULLMUTEX, NULL);
if(ret != SQLITE_OK){
printf("sqlite_open_v2 err\n");
return -1;
}
//检查数据表是否存在,不存在回调里创建
snprintf(sql,sizeof(sql),"select count(*) from sqlite_master where type='table' and name='%s'","tab_product");
ret = sqlite3_exec(g_db, sql, create_callback, g_db, &err);
if(ret != SQLITE_OK){
sqlite3_free(err);
sqlite3_close_v2(g_db);
}
//保存100次数据,正常情况下单次保存;
struct dr_speed_t speed;
time_t t = time(NULL);
for(int i = 0; i < 100; ++i){
memset(&speed, 0, sizeof(struct dr_speed_t));
speed.speed = i;
speed.start_time = t/1000 + i;
speed.state = ((i % 2) == 0) ? 0 : 1;
SqlSave(speed.start_time,&speed,sizeof(struct dr_speed_t));
}
sleep(1);
//批量查询
SqlQuerySpeed(0, time(NULL));
sqlite3_close_v2(g_db);
return 0;
}
开放原子开发者工作坊旨在鼓励更多人参与开源活动,与志同道合的开发者们相互交流开发经验、分享开发心得、获取前沿技术趋势。工作坊有多种形式的开发者活动,如meetup、训练营等,主打技术交流,干货满满,真诚地邀请各位开发者共同参与!
更多推荐
所有评论(0)