C++操作SQLite简明教程

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; }