本節(jié)我們繼續(xù)來學(xué)習(xí)Android數(shù)據(jù)存儲(chǔ)與訪問的第三種方式:SQLite數(shù)據(jù)庫,和其他的SQL數(shù)據(jù)庫不同, 我們并不需要在手機(jī)上另外安裝一個(gè)數(shù)據(jù)庫軟件,Android系統(tǒng)已經(jīng)集成了這個(gè)數(shù)據(jù)庫,我們無需像 使用其他數(shù)據(jù)庫軟件(Oracle,MSSQL,MySql等)又要安裝,然后完成相關(guān)配置,又要改端口之類的! 引言就說這么多,接下來我們來學(xué)習(xí)下這個(gè)東西~
答:下面請(qǐng)聽小豬娓娓道來:
①SQLite是一個(gè)輕量級(jí)的關(guān)系型數(shù)據(jù)庫,運(yùn)算速度快,占用資源少,很適合在移動(dòng)設(shè)備上使用, 不僅支持標(biāo)準(zhǔn)SQL語法,還遵循ACID(數(shù)據(jù)庫事務(wù))原則,無需賬號(hào),使用起來非常方便!
②前面我們學(xué)習(xí)了使用文件與SharedPreference來保存數(shù)據(jù),但是在很多情況下, 文件并不一定是有效的,如多線程并發(fā)訪問是相關(guān)的;app要處理可能變化的復(fù)雜數(shù)據(jù)結(jié)構(gòu)等等! 比如銀行的存錢與取錢!使用前兩者就會(huì)顯得很無力或者繁瑣,數(shù)據(jù)庫的出現(xiàn)可以解決這種問題, 而Android又給我們提供了這樣一個(gè)輕量級(jí)的SQLite,為何不用?
③SQLite支持五種數(shù)據(jù)類型:NULL,INTEGER,REAL(浮點(diǎn)數(shù)),TEXT(字符串文本)和BLOB(二進(jìn)制對(duì)象) 雖然只有五種,但是對(duì)于varchar,char等其他數(shù)據(jù)類型都是可以保存的;因?yàn)镾QLite有個(gè)最大的特點(diǎn): 你可以各種數(shù)據(jù)類型的數(shù)據(jù)保存到任何字段中而不用關(guān)心字段聲明的數(shù)據(jù)類型是什么,比如你 可以在Integer類型的字段中存放字符串,當(dāng)然除了聲明為主鍵INTEGER PRIMARY KEY的字段只能夠存儲(chǔ)64位整數(shù)! 另外, SQLite 在解析CREATE TABLE 語句時(shí), 會(huì)忽略 CREATE TABLE 語句中跟在字段名后面的數(shù)據(jù)類型信息如下面語句會(huì)忽略 name字段的類型信息: CREATE TABLE person (personid integer primary key autoincrement, name varchar(20))
小結(jié)下特點(diǎn):
SQlite通過文件來保存數(shù)據(jù)庫,一個(gè)文件就是一個(gè)數(shù)據(jù)庫,數(shù)據(jù)庫中又包含多個(gè)表格,表格里又有 多條記錄,每個(gè)記錄由多個(gè)字段構(gòu)成,每個(gè)字段有對(duì)應(yīng)的值,每個(gè)值我們可以指定類型,也可以不指定 類型(主鍵除外)
PS:對(duì)了,Android內(nèi)置的SQLite是SQLite 3版本的~
嘿嘿,學(xué)習(xí)一些新東西的時(shí)候,最不喜歡的莫過于遇到一些新名詞,是吧,我們先來說下幾個(gè) 我們?cè)谑褂脭?shù)據(jù)庫時(shí)用到的三個(gè)類:
- SQLiteOpenHelper:抽象類,我們通過繼承該類,然后重寫數(shù)據(jù)庫創(chuàng)建以及更新的方法, 我們還可以通過該類的對(duì)象獲得數(shù)據(jù)庫實(shí)例,或者關(guān)閉數(shù)據(jù)庫!
- SQLiteDatabase:數(shù)據(jù)庫訪問類:我們可以通過該類的對(duì)象來對(duì)數(shù)據(jù)庫做一些增刪改查的操作
- Cursor:游標(biāo),有點(diǎn)類似于JDBC里的resultset,結(jié)果集!可以簡(jiǎn)單理解為指向數(shù)據(jù)庫中某 一個(gè)記錄的指針!
對(duì)于涉及數(shù)據(jù)庫的app,我們不可能手動(dòng)地去給他創(chuàng)建數(shù)據(jù)庫文件,所以需要在第一次啟用app 的時(shí)候就創(chuàng)建好數(shù)據(jù)庫表;而當(dāng)我們的應(yīng)用進(jìn)行升級(jí)需要修改數(shù)據(jù)庫表的結(jié)構(gòu)時(shí),這個(gè)時(shí)候就需要 對(duì)數(shù)據(jù)庫表進(jìn)行更新了;對(duì)于這兩個(gè)操作,安卓給我們提供了SQLiteOpenHelper的兩個(gè)方法, onCreate( )與onUpgrade( )來實(shí)現(xiàn)
方法解析:
- onCreate(database):首次使用軟件時(shí)生成數(shù)據(jù)庫表
- onUpgrade(database,oldVersion,newVersion):在數(shù)據(jù)庫的版本發(fā)生變化時(shí)會(huì)被調(diào)用, 一般在軟件升級(jí)時(shí)才需改變版本號(hào),而數(shù)據(jù)庫的版本是由程序員控制的,假設(shè)數(shù)據(jù)庫現(xiàn)在的 版本是1,由于業(yè)務(wù)的變更,修改了數(shù)據(jù)庫表結(jié)構(gòu),這時(shí)候就需要升級(jí)軟件,升級(jí)軟件時(shí)希望 更新用戶手機(jī)里的數(shù)據(jù)庫表結(jié)構(gòu),為了實(shí)現(xiàn)這一目的,可以把原來的數(shù)據(jù)庫版本設(shè)置為2 或者其他與舊版本號(hào)不同的數(shù)字即可!
代碼示例:
public class MyDBOpenHelper extends SQLiteOpenHelper {
public MyDBOpenHelper(Context context, String name, CursorFactory factory,
int version) {super(context, "my.db", null, 1); }
@Override
//數(shù)據(jù)庫第一次創(chuàng)建時(shí)被調(diào)用
public void onCreate(SQLiteDatabase db) {
db.execSQL("CREATE TABLE person(personid INTEGER PRIMARY KEY AUTOINCREMENT,name VARCHAR(20))");
}
//軟件版本號(hào)發(fā)生改變時(shí)調(diào)用
@Override
public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {
db.execSQL("ALTER TABLE person ADD phone VARCHAR(12) NULL");
}
}
代碼解析:
上述代碼第一次啟動(dòng)應(yīng)用,我們會(huì)創(chuàng)建這個(gè)my.db的文件,并且會(huì)執(zhí)行onCreate()里的方法, 創(chuàng)建一個(gè)Person的表,他又兩個(gè)字段,主鍵personId和name字段;接著如我我們修改db的版本 號(hào),那么下次啟動(dòng)就會(huì)調(diào)用onUpgrade()里的方法,往表中再插入一個(gè)字段!另外這里是插入 一個(gè)字段,所以數(shù)據(jù)不會(huì)丟失,如果是重建表的話,表中的數(shù)據(jù)會(huì)全部丟失,下一節(jié)我們會(huì) 來教大家如何解決這個(gè)問題!
流程小結(jié):
- Step 1:自定義一個(gè)類繼承SQLiteOpenHelper類
- Step 2:在該類的構(gòu)造方法的super中設(shè)置好要?jiǎng)?chuàng)建的數(shù)據(jù)庫名,版本號(hào)
- Step 3:重寫onCreate( )方法創(chuàng)建表結(jié)構(gòu)
- Step 4:重寫onUpgrade( )方法定義版本號(hào)發(fā)生改變后執(zhí)行的操作
當(dāng)我們調(diào)用上面的MyDBOpenhelper的對(duì)象的getWritableDatabase()就會(huì)在下述目錄下創(chuàng)建我們的db 數(shù)據(jù)庫文件:
我們發(fā)現(xiàn)數(shù)據(jù)庫有兩個(gè),前者是我們創(chuàng)建的數(shù)據(jù)庫,而后者則是為了能讓數(shù)據(jù)庫支持事務(wù)而產(chǎn)生的 臨時(shí)的日志文件!一般的大小是0字節(jié)! 而在File Explorer里我們確是打不開文件的,連txt都打不開,何況是.db! 所以下面給大家兩條路選:
- 1.先導(dǎo)出來,然后用SQLite的圖形化工具查看
- 2.配置adb環(huán)境變量后,通過adb shell來查看(命令行,裝比利器)!
嗯,接著給大家演示上述兩種方法,選自己喜歡的一種就可以了~~
這類軟件有很多,筆者用的是SQLite Expert Professional,當(dāng)然你也可以使用其他工具 又需要的可以下載:SQLiteExpert.zip
把我們的db文件導(dǎo)出到電腦桌面,打開SQLiteExpert,界面如下:
別問我怎么玩,導(dǎo)入db后自己慢慢玩,用法很簡(jiǎn)單,不懂百度~
至于方法二,本來是想試試的,后來發(fā)現(xiàn)sqlite命令找不到,試了幾次就算了, 后面用到在細(xì)扣,有興趣可以找下郭霖的《第一行代碼——Android》按著流程圖試試! 這里只貼前面的一部分,命令部分自己看書!
1.配置SDK環(huán)境變量:
右鍵我的電腦 ——> 高級(jí)系統(tǒng)設(shè)置 -> 環(huán)境變量 -> 新建系統(tǒng)變量 -> 把SDK的platform-tools路徑拷貝下: 比如筆者的:C:\Software\Coding\android-sdks-as\platform-tools
確定,然后再找到Path的環(huán)境變量,編輯,然后在結(jié)尾加上:%SDK_HOME%;
然后打開命令行,輸入adb,唰唰唰一堆東西,就說明配置成功了!
——————重點(diǎn)——————: 在執(zhí)行后續(xù)命令行指令之前,針對(duì)你的測(cè)試的機(jī)器可能有幾種: 1.原生模擬器:那行,你跳過這里,繼續(xù)往下 2.Genymotion模擬器:沒戲,Genymotion Shell執(zhí)行不了下述命令 3.真機(jī)(已root):那么你打開File Explorer看看data/data/目錄下有東西沒?沒么? 下面提供一個(gè)方法,就是先裝個(gè)RE文件管理器,然后授予RE Root權(quán)限,接著來到根目錄: 然后長按data目錄,會(huì)彈出這樣的對(duì)話框:
接著等他慢慢修改權(quán)限,修改完畢后,我們?cè)俅未蜷_DDMS的File Explorer,我們可以看到:
好的,可以看到data/data里的東西了! ——————————————————————
2.進(jìn)入adb shell,接著鍵入下述指令,來到我們app的databases目錄下:
接著依次輸入下述指令:
- sqlite3 my.db :打開數(shù)據(jù)庫文件
- .table 查看數(shù)據(jù)庫中有哪些表 接著你直接輸入數(shù)據(jù)庫語句就可以了,比如查詢:Select * from person
- .schema:查看建表語句
- .quit:退出數(shù)據(jù)庫的編輯
- .exit:退出設(shè)備控制臺(tái)
...因?yàn)閟ystem/bin/sh sqlite3: not found,這個(gè)問題,后面Sqlite命令的都用不了, 要看效果圖就自行查詢郭大俠的書吧~而下面我們還是先導(dǎo)出db文件,然后用圖形化的 數(shù)據(jù)庫工具來查看!
假如你沒學(xué)過數(shù)據(jù)庫相關(guān)的語法,或者你懶,不想寫數(shù)據(jù)庫語法,就可以使用Android給我們 提供的操作數(shù)據(jù)庫的一些API方法,下面我們寫個(gè)簡(jiǎn)單的例子來掩飾下這些API的用法!
代碼示例:
運(yùn)行效果圖:
實(shí)現(xiàn)代碼:
布局過于簡(jiǎn)單,就四個(gè)Button,就不貼了,直接貼MainActivity.java的代碼:
public class MainActivity extends AppCompatActivity implements View.OnClickListener {
private Context mContext;
private Button btn_insert;
private Button btn_query;
private Button btn_update;
private Button btn_delete;
private SQLiteDatabase db;
private MyDBOpenHelper myDBHelper;
private StringBuilder sb;
private int i = 1;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
mContext = MainActivity.this;
myDBHelper = new MyDBOpenHelper(mContext, "my.db", null, 1);
bindViews();
}
private void bindViews() {
btn_insert = (Button) findViewById(R.id.btn_insert);
btn_query = (Button) findViewById(R.id.btn_query);
btn_update = (Button) findViewById(R.id.btn_update);
btn_delete = (Button) findViewById(R.id.btn_delete);
btn_query.setOnClickListener(this);
btn_insert.setOnClickListener(this);
btn_update.setOnClickListener(this);
btn_delete.setOnClickListener(this);
}
@Override
public void onClick(View v) {
db = myDBHelper.getWritableDatabase();
switch (v.getId()) {
case R.id.btn_insert:
ContentValues values1 = new ContentValues();
values1.put("name", "呵呵~" + i);
i++;
//參數(shù)依次是:表名,強(qiáng)行插入null值得數(shù)據(jù)列的列名,一行記錄的數(shù)據(jù)
db.insert("person", null, values1);
Toast.makeText(mContext, "插入完畢~", Toast.LENGTH_SHORT).show();
break;
case R.id.btn_query:
sb = new StringBuilder();
//參數(shù)依次是:表名,列名,where約束條件,where中占位符提供具體的值,指定group by的列,進(jìn)一步約束
//指定查詢結(jié)果的排序方式
Cursor cursor = db.query("person", null, null, null, null, null, null);
if (cursor.moveToFirst()) {
do {
int pid = cursor.getInt(cursor.getColumnIndex("personid"));
String name = cursor.getString(cursor.getColumnIndex("name"));
sb.append("id:" + pid + ":" + name + "\n");
} while (cursor.moveToNext());
}
cursor.close();
Toast.makeText(mContext, sb.toString(), Toast.LENGTH_SHORT).show();
break;
case R.id.btn_update:
ContentValues values2 = new ContentValues();
values2.put("name", "嘻嘻~");
//參數(shù)依次是表名,修改后的值,where條件,以及約束,如果不指定三四兩個(gè)參數(shù),會(huì)更改所有行
db.update("person", values2, "name = ?", new String[]{"呵呵~2"});
break;
case R.id.btn_delete:
//參數(shù)依次是表名,以及where條件與約束
db.delete("person", "personid = ?", new String[]{"3"});
break;
}
}
}
當(dāng)然,你可能已經(jīng)學(xué)過SQL,會(huì)寫相關(guān)的SQL語句,而且不想用Android提供的這些API, 你可以直接使用SQLiteDatabase給我們提供的相關(guān)方法:
- execSQL(SQL,Object[]):使用帶占位符的SQL語句,這個(gè)是執(zhí)行修改數(shù)據(jù)庫內(nèi)容的sql語句用的
- rawQuery(SQL,Object[]):使用帶占位符的SQL查詢操作 另外前面忘了介紹下Curosr這個(gè)東西以及相關(guān)屬性,這里補(bǔ)充下: ——Cursor對(duì)象有點(diǎn)類似于JDBC中的ResultSet,結(jié)果集!使用差不多,提供一下方法移動(dòng)查詢結(jié)果的記錄指針:
- move(offset):指定向上或者向下移動(dòng)的行數(shù),整數(shù)表示向下移動(dòng);負(fù)數(shù)表示向上移動(dòng)!
- moveToFirst():指針移動(dòng)到第一行,成功返回true,也說明有數(shù)據(jù)
- moveToLast():指針移動(dòng)到最后一樣,成功返回true;
- moveToNext():指針移動(dòng)到下一行,成功返回true,表明還有元素!
- moveToPrevious():移動(dòng)到上一條記錄
- getCount( )獲得總得數(shù)據(jù)條數(shù)
- isFirst():是否為第一條記錄
- isLast():是否為最后一項(xiàng)
- moveToPosition(int):移動(dòng)到指定行
使用代碼示例:
public void save(Person p)
{
SQLiteDatabase db = dbOpenHelper.getWritableDatabase();
db.execSQL("INSERT INTO person(name,phone) values(?,?)",
new String[]{p.getName(),p.getPhone()});
}
public void delete(Integer id)
{
SQLiteDatabase db = dbOpenHelper.getWritableDatabase();
db.execSQL("DELETE FROM person WHERE personid = ?",
new String[]{id});
}
public void update(Person p)
{
SQLiteDatabase db = dbOpenHelper.getWritableDatabase();
db.execSQL("UPDATE person SET name = ?,phone = ? WHERE personid = ?",
new String[]{p.getName(),p.getPhone(),p.getId()});
}
public Person find(Integer id)
{
SQLiteDatabase db = dbOpenHelper.getReadableDatabase();
Cursor cursor = db.rawQuery("SELECT * FROM person WHERE personid = ?",
new String[]{id.toString()});
//存在數(shù)據(jù)才返回true
if(cursor.moveToFirst())
{
int personid = cursor.getInt(cursor.getColumnIndex("personid"));
String name = cursor.getString(cursor.getColumnIndex("name"));
String phone = cursor.getString(cursor.getColumnIndex("phone"));
return new Person(personid,name,phone);
}
cursor.close();
return null;
}
public List<Person> getScrollData(int offset,int maxResult)
{
List<Person> person = new ArrayList<Person>();
SQLiteDatabase db = dbOpenHelper.getReadableDatabase();
Cursor cursor = db.rawQuery("SELECT * FROM person ORDER BY personid ASC LIMIT= ?,?",
new String[]{String.valueOf(offset),String.valueOf(maxResult)});
while(cursor.moveToNext())
{
int personid = cursor.getInt(cursor.getColumnIndex("personid"));
String name = cursor.getString(cursor.getColumnIndex("name"));
String phone = cursor.getString(cursor.getColumnIndex("phone"));
person.add(new Person(personid,name,phone)) ;
}
cursor.close();
return person;
}
public long getCount()
{
SQLiteDatabase db = dbOpenHelper.getReadableDatabase();
Cursor cursor = db.rawQuery("SELECT COUNT (*) FROM person",null);
cursor.moveToFirst();
long result = cursor.getLong(0);
cursor.close();
return result;
}
PS:除了上面獲取條數(shù)的方法外還可以使用cursor.getCount()方法獲得數(shù)據(jù)的條數(shù), 但是SQL語句要改改!比如*SELECT FROM person;**
本節(jié)給大家介紹了Android內(nèi)置SQLite的基本用法,還是比較簡(jiǎn)單的,下一節(jié)再來研究點(diǎn)稍微 高級(jí)一點(diǎn)的東西,SQLite事務(wù),應(yīng)用更新數(shù)據(jù)庫里數(shù)據(jù)怎么處理,以及數(shù)據(jù)庫存儲(chǔ)大二進(jìn)制文件 的方法!好的,本節(jié)就到這里~
更多建議: