在SQLite的使用过程中最常发生的数据库异常便是数据库被锁定了(SQLITE_BUSY或者SQLITE_LOCKED)。SQLite对于并发的处理机制是允许同一个进程的多个线程同时读取一个数据库,但是任何时刻只允许一个线程/进程写入数据库。所以必须要必须要对数据库的读写进行控制。 SQLite数据库本身提供了两个函数用来处理锁定情况: int sqlite3_busy_handler(sqlite3*, int(*)(void*,int), void*); int sqlite3_busy_timeout(sqlite3*, int ms); 使用这两个函数可以设定当发生数据库锁定的时候,调用什么函数来处理,以及设定锁定时的等待时间。 当然通常情况下我们还可以用另外一种方法来解决这类问题,那便是设置互斥量。一般情况下我们可以使用互斥量,临界区等来实现。在这里我使用了互斥量,主要是因为我需要在多个不同的进程中并发的访问同一个数据库。用于锁定和解锁的函数如下: void Lock() { if((hCounter = OpenMutex(MUTEX_ALL_ACCESS,FALSE,"kangxiaofang")) == NULL) { //如果没有其他进程创建这个互斥量,则重新创建 hCounter = CreateMutex(NULL,FALSE,"kangxiaofang"); WaitForSingleObject(hCounter,INFINITE); } else { WaitForSingleObject(hCounter,INFINITE); } } void UnLock() { //释放使用权 ReleaseMutex(hCounter); //关闭句柄 CloseHandle(hCounter); } 当然我们也可以把等待时间设定为60秒,以免出现死等的现象。具体使用时可以按如下调用: //锁定数据库 Lock(); rc = sqlite3_exec(db, "BEGIN;", 0, 0, &zErrMsg); rc = sqlite3_exec(db, exec, 0, 0, &zErrMsg); rc = sqlite3_exec(db, "COMMIT;", 0, 0, &zErrMsg); //数据库解锁 UnLock(); 经过测试这种方法的消耗要小于循环打开数据库的操作:)接下来我们要讨论一下,如何在SQLite中存取大容量的数据(二进制)。
SQLite包含了如下时间/日期函数:
datetime().......................产生日期和时间
date()...........................产生日期
time()...........................产生时间
strftime().......................对以上三个函数产生的日期和时间进行格式化
datetime()的用法是:datetime(日期/时间,修正符,修正符...)
date()和time()的语法与datetime()相同。
在时间/日期函数里可以使用如下格式的字符串作为参数:
YYYY-MM-DD
YYYY-MM-DD HH:MM
YYYY-MM-DD HH:MM:SS
YYYY-MM-DD HH:MM:SS.SSS
HH:MM
HH:MM:SS
HH:MM:SS.SSS
now
其中now是产生现在的时间。
举例(写这个笔记的时间是2006年10月17日晚8点到10点,测试环境:SQLite 2.8.17,WinXP,北京时间):