大家好,今天來介紹一下SQLite的相關(guān)知識,并結(jié)合Java實現(xiàn)對SQLite數(shù)據(jù)庫的操作。
SQLite是D.Richard Hipp用C語言編寫的開源嵌入式數(shù)據(jù)庫引擎。它支持大多數(shù)的SQL92標(biāo)準(zhǔn),并且可以在所有主要的操作系統(tǒng)上運行。
SQLite由以下幾個部分組成:SQL編譯器、內(nèi)核、后端以及附件。SQLite通過利用虛擬機和虛擬數(shù)據(jù)庫引擎(VDBE),是調(diào)試、修改和擴展SQLite的內(nèi)核變得更加方便。所有SQL語句都被編譯成易讀的、可以在SQLite虛擬機中執(zhí)行的程序集。SQLite的整體結(jié)構(gòu)圖如下:
值得一提的是,袖珍型的SQLite竟然可以支持高達2TB大小的數(shù)據(jù)庫,每個數(shù)據(jù)庫都是以單個文件的形式存在,這些數(shù)據(jù)都是以B-Tree的數(shù)據(jù)結(jié)構(gòu)形式存儲在磁盤上。
在事務(wù)處理方面,SQLite通過數(shù)據(jù)庫級上的獨占性和共享鎖來實現(xiàn)獨立事務(wù)處理。這意味著多個進程可以在同一時間從同一數(shù)據(jù)庫讀取數(shù)據(jù),但只有一個可以寫入數(shù)據(jù)。在某個進程或線程想數(shù)據(jù)庫執(zhí)行寫操作之前,必須獲得獨占鎖。在獲得獨占鎖之后,其他的讀或?qū)懖僮鲗⒉粫侔l(fā)生。
SQLite采用動態(tài)數(shù)據(jù)類型,當(dāng)某個值插入到數(shù)據(jù)庫時,SQLite將會檢查它的類型,如果該類型與關(guān)聯(lián)的列不匹配,SQLite則會嘗試將該值轉(zhuǎn)換成該列的類型,如果不能轉(zhuǎn)換,則該值將作為本身的類型存儲,SQLite稱這為“弱類型”。但有一個特例,如果是INTEGER PRIMARY KEY,則其他類型不會被轉(zhuǎn)換,會報一個“datatype missmatch”的錯誤。
概括來講,SQLite支持NULL、INTEGER、REAL、TEXT和BLOB數(shù)據(jù)類型,分別代表空值、整型值、浮點值、字符串文本、二進制對象。
下面,我們就來親自操作一下SQLite數(shù)據(jù)庫。
在操作之前,朋友們要先下載SQLite數(shù)據(jù)庫,官方的下載頁面是http://sqlite.org/download.html,我是在Windows下試驗,所以我選擇了Precompiled Binaries For Windows下面的sqlite-shell-win32-x86和sqlite-analyzer-win32-x86的zip包,前者是SQLite數(shù)據(jù)庫引擎,后者是SQLite數(shù)據(jù)庫分析器,主要用于分析數(shù)據(jù)庫的狀態(tài)等信息,大家也可以根據(jù)自己的情況去下載。下載完成后分別解壓,得到兩個可執(zhí)行文件,如圖:
這兩個文件可以根據(jù)自己的喜好放置在指定的位置,我將其放在D盤根目錄下。下面我們就來一步一步操作SQLite:
創(chuàng)建數(shù)據(jù)庫:
- D:\>sqlite3 test.db
- SQLite version 3.7.7.1 2011-06-28 17:39:05
- Enter ".help" for instructions
- Enter SQL statements terminated with a ";"
- sqlite> .databases
- seq name file
-
-
-
- 0 main D:\test.db
-
- sqlite>
我們執(zhí)行了sqlite3命令,參數(shù)就是數(shù)據(jù)庫的名稱,如果該數(shù)據(jù)庫已存在,則使用,如果不存在,則新建一個,這里我們簡單的在當(dāng)前位置創(chuàng)建了test.db,你也可以在任何存在的并且可寫的目錄下創(chuàng)建自己的數(shù)據(jù)庫。(如果對于SQLite的命令不太熟悉,可以執(zhí)行“.help”命令列出所有的命令清單進行查看)。
創(chuàng)建表:
- sqlite> CREATE TABLE person (id INTEGER PRIMARY KEY AUTOINCREMENT, name VARCHAR(20), age SMALLINT);
- sqlite> .tables
- person
- sqlite> .schema person
- CREATE TABLE person (id INTEGER PRIMARY KEY AUTOINCREMENT, name VARCHAR(20), age SMALLINT);
- sqlite>
在我們創(chuàng)建表之后,可以用“.tables”命令去查看已有的表,用“.schema”命令去查看表的結(jié)構(gòu),如果后面沒有表名做參數(shù),則將會輸出所有表的建表語句。
插入數(shù)據(jù):
- sqlite> INSERT INTO person VALUES (NULL, 'john', 30);
- sqlite> SELECT * FROM person;
- 1|john|30
從.sql文件導(dǎo)入數(shù)據(jù):
- sqlite> .read test.sql
- sqlite> SELECT * FROM person;
- 1|john|30
- 2|david|35
- 3|henry|40
- sqlite>
分析數(shù)據(jù)庫使用狀態(tài):
- D:\>sqlite3_analyzer test.db
- /** Disk-Space Utilization Report For test.db
-
- Page size in bytes.................... 1024
- Pages in the whole file (measured).... 4
- Pages in the whole file (calculated).. 4
- Pages that store data................. 4 100.0%
- Pages on the freelist (per header).... 0 0.0%
- Pages on the freelist (calculated).... 0 0.0%
- Pages of auto-vacuum overhead......... 0 0.0%
- Number of tables in the database...... 4
- Number of indices..................... 0
- Number of named indices............... 0
- Automatically generated indices....... 0
- Size of the file in bytes............. 4096
- Bytes of user payload stored.......... 39 0.95%
- ...
備份數(shù)據(jù)庫:
備份 SQLite 數(shù)據(jù)庫有兩種方法。如果數(shù)據(jù)庫正在使用中,則應(yīng)從命令行界面使用 .dump 命令。這樣可以創(chuàng)建一個包含必要命令和數(shù)據(jù)的文件,從而重新創(chuàng)建數(shù)據(jù)庫。.dump 命令也可以用于備份數(shù)據(jù)庫表。
- sqlite> .dump
- PRAGMA foreign_keys=OFF;
- BEGIN TRANSACTION;
- CREATE TABLE person (id INTEGER PRIMARY KEY AUTOINCREMENT, name VARCHAR(20), age SMALLINT);
- INSERT INTO "person" VALUES(1,'john',30);
- INSERT INTO "person" VALUES(2,'david',35);
- INSERT INTO "person" VALUES(3,'henry',40);
- DELETE FROM sqlite_sequence;
- INSERT INTO "sqlite_sequence" VALUES('person',3);
- COMMIT;
- sqlite> .output dump.sql
- sqlite> .dump
- sqlite>
我們可以指定輸出的目標(biāo)為一個文件,然后再使用命令時,輸出信息就會寫入指定的文件,如果想恢復(fù)為標(biāo)準(zhǔn)輸出,可以這樣設(shè)定:
- sqlite> .output stdout
- sqlite> .dump
- PRAGMA foreign_keys=OFF;
- BEGIN TRANSACTION;
- CREATE TABLE person (id INTEGER PRIMARY KEY AUTOINCREMENT, name VARCHAR(20), age SMALLINT);
- INSERT INTO "person" VALUES(1,'john',30);
- INSERT INTO "person" VALUES(2,'david',35);
- INSERT INTO "person" VALUES(3,'henry',40);
- DELETE FROM sqlite_sequence;
- INSERT INTO "sqlite_sequence" VALUES('person',3);
- COMMIT;
- sqlite>
如果數(shù)據(jù)庫沒有處于使用狀態(tài),則可以直接將數(shù)據(jù)庫文件復(fù)制到安全位置。
最后,我們可以使用“.quit”或“.exit”退出SQLite。
管理工具:
現(xiàn)在網(wǎng)絡(luò)上的SQLite管理工具很多,我向大家推薦一款好用的工具:SQLite Expert。
在Java中使用SQLite:
我們要想在Java中使用SQLite,需要下載SQLite相關(guān)驅(qū)動,推薦大家到http://www.xerial.org/trac/Xerial/wiki/SQLiteJDBC頁面去下載最新的驅(qū)動包,現(xiàn)在最新版本是sqlite-jdbc-3.7.2.jar,體積有點大,因為它包含了Linux、Mac、Windows的本地類庫,如圖:
下載了驅(qū)動之后,我們新建一個項目,名為sqlite:
在上圖中,我們引入sqlite驅(qū)動包到類路徑下,然后建立一個db的文件夾,用于放置數(shù)據(jù)庫文件。最后我們看一下Test.java代碼:
- package com.scott.sqlite;
- import java.sql.Connection;
- import java.sql.DriverManager;
- import java.sql.ResultSet;
- import java.sql.Statement;
-
- public class Test {
- public static void main(String[] args) throws Exception {
- Class.forName("org.sqlite.JDBC");
- Connection conn = DriverManager.getConnection("jdbc:sqlite:db/test.db");
- Statement stmt = conn.createStatement();
-
- stmt.executeUpdate("DROP TABLE IF EXISTS person");
- stmt.executeUpdate("CREATE TABLE person(id INTEGER, name STRING)");
- stmt.executeUpdate("INSERT INTO person VALUES(1, 'john')");
- stmt.executeUpdate("INSERT INTO person VALUES(2, 'david')");
- stmt.executeUpdate("INSERT INTO person VALUES(3, 'henry')");
- ResultSet rs = stmt.executeQuery("SELECT * FROM person");
- while (rs.next()) {
- System.out.println("id=>" + rs.getInt("id") + ", name=>" + rs.getString("name"));
- }
- stmt.close();
- conn.close();
- }
- }
執(zhí)行Test.java文件,結(jié)果如下:
這個時候,在我們的db目錄下,就生成了一個test.db的文件:
SQLite使用須知:
目前沒有可用于 SQLite 的網(wǎng)絡(luò)服務(wù)器。從應(yīng)用程序運行位于其他計算機上的 SQLite 的惟一方法是從網(wǎng)絡(luò)共享運行。這樣會導(dǎo)致一些問題,像 UNIX? 和 Windows? 網(wǎng)絡(luò)共享都存在文件鎖定問題。還有由于與訪問網(wǎng)絡(luò)共享相關(guān)的延遲而帶來的性能下降問題。
SQLite 只提供數(shù)據(jù)庫級的鎖定。
SQLite 沒有用戶帳戶概念,而是根據(jù)文件系統(tǒng)確定所有數(shù)據(jù)庫的權(quán)限。