C++操作SQLite简明教程

所属分类: 软件编程 / C 语言 阅读数: 151
收藏 0 赞 0 分享

SQLite是一款轻型的本地文件数据库,是遵守ACID的关联式数据库管理系统。它的设计目标是嵌入式的,而且目前已经在很多嵌入式产品中使用了它,它的功能强、速度快,它占用资源非常的低,在嵌入式设备中,可能只需要几百K的内存就够了。它能够支持Windows/Linux/Unix等主流的操作系统,同时能够跟很多程序语言相结合。

一、SQLite的数据类型

在进行数据库操作之前,有个问题需要说明,就是SQLite的数据类型,和其他的数据库不同,Sqlite支持的数据类型有他自己的特色:Typelessness(无类型)。 SQLite是无类型的,这意味着你可以保存任何类型的数据到你所想要保存的任何表的任何列中, 无论这列声明的数据类型是什么。

而大多数的数据库在数据类型上都有严格的限制,在建立表的时候,每一列都必须制定一个数据类型,只有符合该数据类型的数据可以被保存在这一列当中。而在SQLite 2.X中,数据类型这个属性只属于数据本生,而不和数据被存在哪一列有关,也就是说数据的类型并不受数据列限制(有一个例外:INTEGER PRIMARY KEY,该列只能存整型数据)。

但是当SQLite进入到3.0版本的时候,这个问题似乎又有了新的答案,SQLite的开发者开始限制这种无类型的使用,在3.0版本当中,每一列开始拥有自己的类型,并且在数据存入该列的时候,数据库会试图把数据的类型向该类型转换,然后以转换之后的类型存储。当然,如果转换被认为是不可行的,SQLite仍然会存储这个数据,就像他的前任版本一样。

举个例子,如果你企图向一个INTEGER类型的列中插入一个字符串,SQLite会检查这个字符串是否有整型数据的特征, 如果有而且可以被数据库所识别,那么该字符串会被转换成整型再保存,如果不行,则还是作为字符串存储。
诚然SQLite允许忽略数据类型, 但是仍然建议在你的Create Table语句中指定数据类型. 因为数据类型对于你和其他的程序员交流, 或者你准备换掉你的数据库引擎时能起到一个提示或帮助的作用. SQLite支持常见的数据类型, 如:

1.NULL,值是NULL
2.INTEGER,值是有符号整形,根据值的大小以1,2,3,4,6或8字节存放
3.REAL,值是浮点型值,以8字节IEEE浮点数存放
4.TEXT,值是文本字符串,使用数据库编码(UTF-8,UTF-16BE或者UTF-16LE)
5.BLOB,只是一个数据块,完全按照输入存放(即没有准换)

SQLite的操作接口

SQLite的2个重要结构体:

sqlite3 *pdb,数据库句柄,跟文件句柄FILE很类似
sqlite3_stmt *stmt,这个相当于ODBC的Command对象,用于保存编译好的SQL语句

SQLite的5个主要的函数:

sqlite3_open(), 打开数据库
sqlite3_exec(),执行非查询的sql语句
sqlite3_prepare(),准备sql语句,执行select语句或者要使用parameter bind时,用这个函数(封装了sqlite3_exec).
sqlite3_step(),在调用sqlite3_prepare后,使用这个函数在记录集中移动。
sqlite3_close(),关闭数据库文件

还有一系列的函数,用于从记录集字段中获取数据,如:

sqlite3_column_text(),取text类型的数据
sqlite3_column_blob(),取blob类型的数据
sqlite3_column_int(),取int类型的数据

C++使用前准备

下载SQLite,分别从http://www.sqlite.org/2013/sqlite-amalgamation-3071700.zip、 http://www.sqlite.org/2013/sqlite-dll-win32-x86-3071700.zip下载SQLite的源文件和库文件。

但是sqlite-dll-win32-x86-3071700.zip中并没有提供SQLite的lib文件,需要自己编译生成。解压sqlite-dll-win32-x86-3071700.zip到sqlite-dll-win32-x86-3071700目录,再将VS安装目录下VC中的LIB.EXE、LINK.EXE、mspdb80.dll(这里用的是VS2008)拷贝到sqlite-dll-win32-x86-3071700,执行LIB.EXE /DEF:SQLITE3.DEF /MACHINE:IX86 就可以得到SQLITE3.LIB文件。

C++访问SQLite

C++对SQLite进行操作的代码如下:

复制代码 代码如下:

#include "stdafx.h"
#include <string.h>
using namespace std;

#include "sqlite3.h"
#pragma comment(lib, "SQLITE3.LIB")

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

int _tmain(int argc, _TCHAR* argv[])
{
 sqlite3 * pDB;
 char* errMsg;

// 打开SQLite数据库
 int res = sqlite3_open("sql.db", &pDB);
 if ( res != SQLITE_OK ){
 printf("Can't open database: %s\n", sqlite3_errmsg(pDB));
 sqlite3_close(pDB);
 return -1;
 }

// 创建表
 string strSQL= "create table test (id int, name text);";
 res = sqlite3_exec(pDB , strSQL.c_str() ,0 ,0, &errMsg);
 if (res != SQLITE_OK)
 {
 printf("Create table error: %s\n", errMsg);
 //return -1;
 }

// 插入数据
 res = sqlite3_exec(pDB,"begin transaction;",0,0, &errMsg);
 for (int i= 1; i < 10; ++i)
 {
 char sql[512];
 sprintf_s(sql, "insert into test values(%d, %s);", (i+10), "\"Test Name\"");

res = sqlite3_exec(pDB, sql,0,0, &errMsg);
 if (res != SQLITE_OK)
 {
 printf("Insert error: %s\n", errMsg);
 return -1;
 }
 }
 res = sqlite3_exec(pDB,"commit transaction;",0,0, &errMsg);

// 查询数据
 strSQL= "select * from test;";
 res = sqlite3_exec(pDB, strSQL.c_str(), SelectCallback, 0 , &errMsg);
 if (res != SQLITE_OK)
 {
 printf("Select error: %s\n", errMsg);
 return -1;
 }

 // 关闭数据库
 sqlite3_close(pDB);

return 0;
}

更多精彩内容其他人还在看

用标准c++实现string与各种类型之间的转换

这个类在头文件中定义, < sstream>库定义了三种类:istringstream、ostringstream和stringstream,分别用来进行流的输入、输出和输入输出操作。另外,每个类都有一个对应的宽字符集版本
收藏 0 赞 0 分享

C++如何通过ostringstream实现任意类型转string

再使用整型转string的时候感觉有点棘手,因为itoa不是标准C里面的,而且即便是有itoa,其他类型转string不是很方便。后来去网上找了一下,发现有一个好方法
收藏 0 赞 0 分享

C/C++指针小结

要搞清一个指针需要搞清指针的四方面的内容:指针的类型,指针所指向的类型,指针的值或者叫指针所指向的内存区,还有指针本身所占据的内存区
收藏 0 赞 0 分享

C++ 类的静态成员深入解析

在C++中类的静态成员变量和静态成员函数是个容易出错的地方,本文先通过几个例子来总结静态成员变量和成员函数使用规则,再给出一个实例来加深印象
收藏 0 赞 0 分享

C++类的静态成员初始化详细讲解

通常静态数据成员在类声明中声明,在包含类方法的文件中初始化.初始化时使用作用域操作符来指出静态成员所属的类.但如果静态成员是整型或是枚举型const,则可以在类声明中初始化
收藏 0 赞 0 分享

C++类静态成员与类静态成员函数详解

静态成员不可在类体内进行赋值,因为它是被所有该类的对象所共享的。你在一个对象里给它赋值,其他对象里的该成员也会发生变化。为了避免混乱,所以不可在类体内进行赋值
收藏 0 赞 0 分享

C++中的friend友元函数详细解析

友元可以是一个函数,该函数被称为友元函数;友元也可以是一个类,该类被称为友元类。友元函数的特点是能够访问类中的私有成员的非成员函数。友元函数从语法上看,它与普通函数一样,即在定义上和调用上与普通函数一样
收藏 0 赞 0 分享

static全局变量与普通的全局变量的区别详细解析

以下是对static全局变量与普通的全局变量的区别进行了详细的分析介绍,需要的朋友可以过来参考下,希望对大家有所帮助
收藏 0 赞 0 分享

C++ explicit关键字的应用方法详细讲解

C++ explicit关键字用来修饰类的构造函数,表明该构造函数是显式的,既然有"显式"那么必然就有"隐式",那么什么是显示而什么又是隐式的呢?下面就让我们一起来看看这方面的知识吧
收藏 0 赞 0 分享

教你5分钟轻松搞定内存字节对齐

随便google一下,人家就可以跟你解释的,一大堆的道理,我们没怎么多时间,讨论为何要对齐.直入主题,怎么判断内存对齐规则,sizeof的结果怎么来的,请牢记以下3条原则
收藏 0 赞 0 分享
查看更多