sqlite3--关于数据库中数组的存取
2014-01-08 13:48
561 查看
从数据库中存取数组有两个方案:1、将数组转换为字符串然后存入到text类字段中,查询时将读出的字符串再按一定规则转为数组
2、将数组直接按二进制数据存入blob类字段中
方法1:数组存blob类字段
首先看一段关于blob类二进制数据的存取的文章
sqlite 操作二进制数据需要用一个辅助的数据类型:sqlite3_stmt *
。
这个数据类型记录了一个“sql语句”。为什么我把“sql语句”用双引号引起来?因为你可以把
sqlite3_stmt * 所表示的内容看成是 sql语句,但是实际上它不是我们所熟知的sql语句。它是一个已经把sql语句解析了的、用sqlite自己标记记录的内部数据结构。
正因为这个结构已经被解析了,所以你可以往这个语句里插入二进制数据。当然,把二进制数据插到 sqlite3_stmt
结构里可不能直接 memcpy
,也不能像 std::string
那样用 + 号。必须用 sqlite
提供的函数来插入。
要插入二进制,前提是这个表的字段的类型是 blob
类型。我假设有这么一张表:
create table Tbl_2( ID integer, file_content
blob )
首先声明
sqlite3_stmt * stat;
然后,把一个 sql
语句解析到 stat 结构里去:
sqlite3_prepare( db,
“insert into Tbl_2( ID, file_content) values( 10, ? )”, -1, &stat, 0 );
上面的函数完成 sql
语句的解析。第一个参数跟前面一样,是个 sqlite3 *
类型变量,第二个参数是一个 sql
语句。
这个 sql
语句特别之处在于 values
里面有个 ? 号。在sqlite3_prepare函数里,?号表示一个未定的值,它的值等下才插入。
第三个参数我写的是-1,这个参数含义是前面 sql
语句的长度。如果小于0,sqlite会自动计算它的长度(把sql语句当成以/0结尾的字符串)。
第四个参数是 sqlite3_stmt
的指针的指针。解析以后的sql语句就放在这个结构里。
第五个参数我也不知道是干什么的。为0就可以了。
如果这个函数执行成功(返回值是 SQLITE_OK
且 stat 不为NULL
),那么下面就可以开始插入二进制数据。
sqlite3_bind_blob( stat, 1, pdata, (int)(length_of_data_in_bytes), NULL );
//pdata为数据缓冲区,length_of_data_in_bytes为数据大小,以字节为单位
这个函数一共有5个参数。
第1个参数:是前面prepare得到的 sqlite3_stmt *
类型变量。
第2个参数:?号的索引。前面prepare的sql语句里有一个?号,假如有多个?号怎么插入?方法就是改变
bind_blob 函数第2个参数。这个参数我写1,表示这里插入的值要替换 stat
的第一个?号(这里的索引从1开始计数,而非从0开始)。如果你有多个?号,就写多个
bind_blob 语句,并改变它们的第2个参数就替换到不同的?号。如果有?号没有替换,sqlite为它取值null。
第3个参数:二进制数据起始指针。
第4个参数:二进制数据的长度,以字节为单位。
第5个参数:是个析够回调函数,告诉sqlite当把数据处理完后调用此函数来析够你的数据。这个参数我还没有使用过,因此理解也不深刻。但是一般都填NULL,需要释放的内存自己用代码来释放。
bind完了之后,二进制数据就进入了你的“sql语句”里了。你现在可以把它保存到数据库里:
int result =
sqlite3_step( stat );
通过这个语句,stat
表示的sql语句就被写到了数据库里。
最后,要把 sqlite3_stmt
结构给释放:
sqlite3_finalize( stat );
//把刚才分配的内容析构掉
跟前面一样,先声明sqlite3_stmt *
类型变量:
sqlite3_stmt * stat;
然后,把一个 sql
语句解析到 stat 结构里去:
sqlite3_prepare( db,
“select * from Tbl_2”, -1, &stat, 0 );
当 prepare成功之后(返回值是
SQLITE_OK),开始查询数据。
int result =
sqlite3_step( stat );
这一句的返回值是SQLITE_ROW时表示成功(不是SQLITE_OK
)。
你可以循环执行sqlite3_step
函数,一次step查询出一条记录。直到返回值不为
SQLITE_ROW时表示查询结束。
然后开始获取第一个字段:ID
的值。ID是个整数,用下面这个语句获取它的值:
int id = sqlite3_column_int( stat, 0 );
//第2个参数表示获取第几个字段内容,从0开始计算,因为我的表的ID字段是第一个字段,因此这里我填0
下面开始获取 file_content
的值,因为 file_content
是二进制,因此我需要得到它的指针,还有它的长度:
const void * pFileContent = sqlite3_column_blob( stat, 1 );
int len =
sqlite3_column_bytes( stat, 1 );
这样就得到了二进制的值。
把 pFileContent的内容保存出来之后,不要忘了释放 sqlite3_stmt
结构:
sqlite3_finalize( stat );
//把刚才分配的内容析构掉
更多关于sqlite 的c语言API函数参考我的其他博文!
下面直接上程序:
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sqlite3.h>
static sqlite3 *db = NULL;
//不使用回调函数的查询
int inquire_nocb(sqlite3 *db)
{ int id;
int len;
int j=0;
unsigned int *shuju;
sqlite3_stmt *stmt;
char *sql;
char *zErrMsg;
sql = "SELECT * FROM lianxiData";
sqlite3_prepare(db, sql, strlen(sql), &stmt, 0);
sqlite3_step(stmt);
id=sqlite3_column_int(stmt,0);
shuju=sqlite3_column_blob(stmt,1);
len=sqlite3_column_bytes(stmt,1);
sqlite3_finalize(stmt);
printf("%d\n",id);
printf("%d\n",len);
for(j=0;j<5;j++)
{
printf(”%d",*shuju);
shuju++;
}
return 0;
}
void db_exec(sqlite3 *db)
{
unsigned int shuju[5]={1212,5555,111,60000,5};
char *sql = "insert into lianxiData (ID,DATE) values(1,?)";
sqlite3_stmt *stmt;
int rc;
//char *name="xiaoliang";
rc = sqlite3_prepare(db, sql, strlen(sql), &stmt, NULL);
if (rc != SQLITE_OK)
{
fprintf(stderr, "sql error:%s\n", sqlite3_errmsg(db));
}
//sqlite3_bind_int(stmt, 1, cout);
sqlite3_bind_blob(stmt, 1, shuju, 20, NULL);
sqlite3_step(stmt);
sqlite3_finalize(stmt);
}
//创建表函数
int createTable(sqlite3 *db)
{
char *zErrMsg = 0;
char *sql = "CREATE TABLE lianxiData (ID INTEGER PRIMARY KEY,DATE BLOB);";
if(SQLITE_OK != sqlite3_exec(db,sql,0,0,&zErrMsg)){
printf("operate failed: %s\n",zErrMsg);
}
}
//删除数据函数
int deleteRecord(sqlite3 *db)
{
char *sql;
char *zErrMsg;
char **azResult=0;
int nrow = 0,ncolumn = 0;
int i;
sql = "DELETE FROM SensorData WHERE ID=4;";
if(SQLITE_OK != sqlite3_exec(db,sql,0,0,&zErrMsg)){
printf("operate failed: %s\n",zErrMsg);
}
sql = "SELECT * FROM SensorData;";
if(SQLITE_OK != sqlite3_get_table(db,sql,&azResult ,&nrow,&ncolumn,&zErrMsg)){
printf("operate failed: %s\n",zErrMsg);
}
printf("row:%d column:%d\n",nrow,ncolumn);
printf("After deleting,the result of querying is :\n");
for(i=0;i<(nrow+1)*ncolumn;i++)
printf("azResult[%d] = %s\n",i,azResult[i]);
sqlite3_free_table(azResult);
}
int main(void)
{
int rc;
//打开指定的数据库文件
rc = sqlite3_open("hongwai1.db",&db);
if(rc){
fprintf(stderr,"can't open database: %s\n",sqlite3_errmsg(db));
sqlite3_close(db);
exit(1);
}else
printf("You have opened a sqlite3 database successfully!\n");
//createTable(db);
createTable(db);
db_exec(db);
inquire_nocb(db);
}
运行结果:
you have opend a splite3 datebase sucselly!
1
20
12125555111600005
方法2:数组转字符串
其中关于数组转字符串的sprint函数和关于插入查询text类段的API函数都参考我的其他博文!
直接上程序:
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sqlite3.h>
static sqlite3 *db = NULL;
//不使用回调函数的查询
int inquire_nocb(sqlite3 *db)
{
int id;
int j=0;
unsigned int shuju[5];
char *zhuanhuan_char;
sqlite3_stmt *stmt;
char *sql;
char *zErrMsg;
sql = "SELECT * FROM lianxiData";
sqlite3_prepare(db, sql, strlen(sql), &stmt, 0);
sqlite3_step(stmt);
id=sqlite3_column_int(stmt,0);
zhuanhuan_char=sqlite3_column_text(stmt,1);
sqlite3_finalize(stmt);
printf("%d\n",id);
for(j=0;j<5;j++)
{
shuju[j]=(*zhuanhuan_char-0x30)*10+(*(++zhuanhuan_char)-0x30);
zhuanhuan_char++;
printf("%d",shuju[j]);
}
return 0;
}
void db_exec(sqlite3 *db)
{ int j=0;
unsigned int shuju[5]={12,55,11,60,33};
char *sql = "insert into lianxiData (ID,DATE) values(1,?)";
sqlite3_stmt *stmt;
char zhuanhuan_char[100]="";
int rc;
//char *name="xiaoliang";
for(j;j<5;j++)
{
sprintf(zhuanhuan_char,"%s%d",zhuanhuan_char,shuju[j]);
}
rc = sqlite3_prepare(db, sql, strlen(sql), &stmt, NULL);
if (rc != SQLITE_OK)
{
fprintf(stderr, "sql error:%s\n", sqlite3_errmsg(db));
}
//sqlite3_bind_int(stmt, 1, cout);
sqlite3_bind_text(stmt, 1, zhuanhuan_char, strlen(zhuanhuan_char), NULL);
sqlite3_step(stmt);
sqlite3_finalize(stmt);
}
//创建表函数
int createTable(sqlite3 *db)
{
char *zErrMsg = 0;
char *sql = "CREATE TABLE lianxiData (ID INTEGER PRIMARY KEY,DATE TEXT);";
if(SQLITE_OK != sqlite3_exec(db,sql,0,0,&zErrMsg)){
printf("operate failed: %s\n",zErrMsg);
}
}
int main(void)
{
int rc;
//打开指定的数据库文件
rc = sqlite3_open("hongwai2.db",&db);
if(rc){
fprintf(stderr,"can't open database: %s\n",sqlite3_errmsg(db));
sqlite3_close(db);
exit(1);
}else
printf("You have opened a sqlite3 database successfully!\n");
//createTable(db);
createTable(db);
db_exec(db);
inquire_nocb(db);
}
2、将数组直接按二进制数据存入blob类字段中
方法1:数组存blob类字段
首先看一段关于blob类二进制数据的存取的文章
sqlite 操作二进制数据需要用一个辅助的数据类型:sqlite3_stmt *
。
这个数据类型记录了一个“sql语句”。为什么我把“sql语句”用双引号引起来?因为你可以把
sqlite3_stmt * 所表示的内容看成是 sql语句,但是实际上它不是我们所熟知的sql语句。它是一个已经把sql语句解析了的、用sqlite自己标记记录的内部数据结构。
正因为这个结构已经被解析了,所以你可以往这个语句里插入二进制数据。当然,把二进制数据插到 sqlite3_stmt
结构里可不能直接 memcpy
,也不能像 std::string
那样用 + 号。必须用 sqlite
提供的函数来插入。
写入二进制
下面说写二进制的步骤。要插入二进制,前提是这个表的字段的类型是 blob
类型。我假设有这么一张表:
create table Tbl_2( ID integer, file_content
blob )
首先声明
sqlite3_stmt * stat;
然后,把一个 sql
语句解析到 stat 结构里去:
sqlite3_prepare( db,
“insert into Tbl_2( ID, file_content) values( 10, ? )”, -1, &stat, 0 );
上面的函数完成 sql
语句的解析。第一个参数跟前面一样,是个 sqlite3 *
类型变量,第二个参数是一个 sql
语句。
这个 sql
语句特别之处在于 values
里面有个 ? 号。在sqlite3_prepare函数里,?号表示一个未定的值,它的值等下才插入。
第三个参数我写的是-1,这个参数含义是前面 sql
语句的长度。如果小于0,sqlite会自动计算它的长度(把sql语句当成以/0结尾的字符串)。
第四个参数是 sqlite3_stmt
的指针的指针。解析以后的sql语句就放在这个结构里。
第五个参数我也不知道是干什么的。为0就可以了。
如果这个函数执行成功(返回值是 SQLITE_OK
且 stat 不为NULL
),那么下面就可以开始插入二进制数据。
sqlite3_bind_blob( stat, 1, pdata, (int)(length_of_data_in_bytes), NULL );
//pdata为数据缓冲区,length_of_data_in_bytes为数据大小,以字节为单位
这个函数一共有5个参数。
第1个参数:是前面prepare得到的 sqlite3_stmt *
类型变量。
第2个参数:?号的索引。前面prepare的sql语句里有一个?号,假如有多个?号怎么插入?方法就是改变
bind_blob 函数第2个参数。这个参数我写1,表示这里插入的值要替换 stat
的第一个?号(这里的索引从1开始计数,而非从0开始)。如果你有多个?号,就写多个
bind_blob 语句,并改变它们的第2个参数就替换到不同的?号。如果有?号没有替换,sqlite为它取值null。
第3个参数:二进制数据起始指针。
第4个参数:二进制数据的长度,以字节为单位。
第5个参数:是个析够回调函数,告诉sqlite当把数据处理完后调用此函数来析够你的数据。这个参数我还没有使用过,因此理解也不深刻。但是一般都填NULL,需要释放的内存自己用代码来释放。
bind完了之后,二进制数据就进入了你的“sql语句”里了。你现在可以把它保存到数据库里:
int result =
sqlite3_step( stat );
通过这个语句,stat
表示的sql语句就被写到了数据库里。
最后,要把 sqlite3_stmt
结构给释放:
sqlite3_finalize( stat );
//把刚才分配的内容析构掉
读出二进制
下面说读二进制的步骤。跟前面一样,先声明sqlite3_stmt *
类型变量:
sqlite3_stmt * stat;
然后,把一个 sql
语句解析到 stat 结构里去:
sqlite3_prepare( db,
“select * from Tbl_2”, -1, &stat, 0 );
当 prepare成功之后(返回值是
SQLITE_OK),开始查询数据。
int result =
sqlite3_step( stat );
这一句的返回值是SQLITE_ROW时表示成功(不是SQLITE_OK
)。
你可以循环执行sqlite3_step
函数,一次step查询出一条记录。直到返回值不为
SQLITE_ROW时表示查询结束。
然后开始获取第一个字段:ID
的值。ID是个整数,用下面这个语句获取它的值:
int id = sqlite3_column_int( stat, 0 );
//第2个参数表示获取第几个字段内容,从0开始计算,因为我的表的ID字段是第一个字段,因此这里我填0
下面开始获取 file_content
的值,因为 file_content
是二进制,因此我需要得到它的指针,还有它的长度:
const void * pFileContent = sqlite3_column_blob( stat, 1 );
int len =
sqlite3_column_bytes( stat, 1 );
这样就得到了二进制的值。
把 pFileContent的内容保存出来之后,不要忘了释放 sqlite3_stmt
结构:
sqlite3_finalize( stat );
//把刚才分配的内容析构掉
更多关于sqlite 的c语言API函数参考我的其他博文!
下面直接上程序:
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sqlite3.h>
static sqlite3 *db = NULL;
//不使用回调函数的查询
int inquire_nocb(sqlite3 *db)
{ int id;
int len;
int j=0;
unsigned int *shuju;
sqlite3_stmt *stmt;
char *sql;
char *zErrMsg;
sql = "SELECT * FROM lianxiData";
sqlite3_prepare(db, sql, strlen(sql), &stmt, 0);
sqlite3_step(stmt);
id=sqlite3_column_int(stmt,0);
shuju=sqlite3_column_blob(stmt,1);
len=sqlite3_column_bytes(stmt,1);
sqlite3_finalize(stmt);
printf("%d\n",id);
printf("%d\n",len);
for(j=0;j<5;j++)
{
printf(”%d",*shuju);
shuju++;
}
return 0;
}
void db_exec(sqlite3 *db)
{
unsigned int shuju[5]={1212,5555,111,60000,5};
char *sql = "insert into lianxiData (ID,DATE) values(1,?)";
sqlite3_stmt *stmt;
int rc;
//char *name="xiaoliang";
rc = sqlite3_prepare(db, sql, strlen(sql), &stmt, NULL);
if (rc != SQLITE_OK)
{
fprintf(stderr, "sql error:%s\n", sqlite3_errmsg(db));
}
//sqlite3_bind_int(stmt, 1, cout);
sqlite3_bind_blob(stmt, 1, shuju, 20, NULL);
sqlite3_step(stmt);
sqlite3_finalize(stmt);
}
//创建表函数
int createTable(sqlite3 *db)
{
char *zErrMsg = 0;
char *sql = "CREATE TABLE lianxiData (ID INTEGER PRIMARY KEY,DATE BLOB);";
if(SQLITE_OK != sqlite3_exec(db,sql,0,0,&zErrMsg)){
printf("operate failed: %s\n",zErrMsg);
}
}
//删除数据函数
int deleteRecord(sqlite3 *db)
{
char *sql;
char *zErrMsg;
char **azResult=0;
int nrow = 0,ncolumn = 0;
int i;
sql = "DELETE FROM SensorData WHERE ID=4;";
if(SQLITE_OK != sqlite3_exec(db,sql,0,0,&zErrMsg)){
printf("operate failed: %s\n",zErrMsg);
}
sql = "SELECT * FROM SensorData;";
if(SQLITE_OK != sqlite3_get_table(db,sql,&azResult ,&nrow,&ncolumn,&zErrMsg)){
printf("operate failed: %s\n",zErrMsg);
}
printf("row:%d column:%d\n",nrow,ncolumn);
printf("After deleting,the result of querying is :\n");
for(i=0;i<(nrow+1)*ncolumn;i++)
printf("azResult[%d] = %s\n",i,azResult[i]);
sqlite3_free_table(azResult);
}
int main(void)
{
int rc;
//打开指定的数据库文件
rc = sqlite3_open("hongwai1.db",&db);
if(rc){
fprintf(stderr,"can't open database: %s\n",sqlite3_errmsg(db));
sqlite3_close(db);
exit(1);
}else
printf("You have opened a sqlite3 database successfully!\n");
//createTable(db);
createTable(db);
db_exec(db);
inquire_nocb(db);
}
运行结果:
you have opend a splite3 datebase sucselly!
1
20
12125555111600005
方法2:数组转字符串
其中关于数组转字符串的sprint函数和关于插入查询text类段的API函数都参考我的其他博文!
直接上程序:
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sqlite3.h>
static sqlite3 *db = NULL;
//不使用回调函数的查询
int inquire_nocb(sqlite3 *db)
{
int id;
int j=0;
unsigned int shuju[5];
char *zhuanhuan_char;
sqlite3_stmt *stmt;
char *sql;
char *zErrMsg;
sql = "SELECT * FROM lianxiData";
sqlite3_prepare(db, sql, strlen(sql), &stmt, 0);
sqlite3_step(stmt);
id=sqlite3_column_int(stmt,0);
zhuanhuan_char=sqlite3_column_text(stmt,1);
sqlite3_finalize(stmt);
printf("%d\n",id);
for(j=0;j<5;j++)
{
shuju[j]=(*zhuanhuan_char-0x30)*10+(*(++zhuanhuan_char)-0x30);
zhuanhuan_char++;
printf("%d",shuju[j]);
}
return 0;
}
void db_exec(sqlite3 *db)
{ int j=0;
unsigned int shuju[5]={12,55,11,60,33};
char *sql = "insert into lianxiData (ID,DATE) values(1,?)";
sqlite3_stmt *stmt;
char zhuanhuan_char[100]="";
int rc;
//char *name="xiaoliang";
for(j;j<5;j++)
{
sprintf(zhuanhuan_char,"%s%d",zhuanhuan_char,shuju[j]);
}
rc = sqlite3_prepare(db, sql, strlen(sql), &stmt, NULL);
if (rc != SQLITE_OK)
{
fprintf(stderr, "sql error:%s\n", sqlite3_errmsg(db));
}
//sqlite3_bind_int(stmt, 1, cout);
sqlite3_bind_text(stmt, 1, zhuanhuan_char, strlen(zhuanhuan_char), NULL);
sqlite3_step(stmt);
sqlite3_finalize(stmt);
}
//创建表函数
int createTable(sqlite3 *db)
{
char *zErrMsg = 0;
char *sql = "CREATE TABLE lianxiData (ID INTEGER PRIMARY KEY,DATE TEXT);";
if(SQLITE_OK != sqlite3_exec(db,sql,0,0,&zErrMsg)){
printf("operate failed: %s\n",zErrMsg);
}
}
int main(void)
{
int rc;
//打开指定的数据库文件
rc = sqlite3_open("hongwai2.db",&db);
if(rc){
fprintf(stderr,"can't open database: %s\n",sqlite3_errmsg(db));
sqlite3_close(db);
exit(1);
}else
printf("You have opened a sqlite3 database successfully!\n");
//createTable(db);
createTable(db);
db_exec(db);
inquire_nocb(db);
}
相关文章推荐
- Android的轻量级数据库sqlite、以及文件存取byte数组
- 关于PostgreSQL数据库的大对象存取
- 关于word文档的数据库存取
- ios数据库sqlite-第三方框架FMDB,关于线程安全的事务处理
- sqlite3数据库存取和取出图片
- 关于Androdi中SQLITE 3采用GBK编码存储,数据库中文乱码问题。
- 关于多线程访问sqlite 数据库的思考
- 关于数据库存取图片的些许事例
- 关于Android SQLite数据库版本升级的补充。。
- 【脚本语言系列】关于Python数据库处理SQLite,你需要知道的事
- 关于android sqlite数据库使用的几点心得
- Android学习中关于SQLite的一个小Demo(数据库的创建、数据的增删查改)
- 关于sqlite3数据库文件格式的分析
- android关于数据库SQLite的简单使用
- sqlite语句,关于使用FMDB操作带有外键的数据库
- 关于sqlite中图片数据的存取问题
- 关于数据库SQLiteDatabase的增删改查四个方法的参数详解
- 关于sqlite3数据库文件格式的分析
- 关于sqlite和wxsqlite的数据库加密
- iOS关于SQLite存取时间数据类型的两条常用方法。