Android Sqlite 數據庫版本更新
http://87426628.blog.163.com/blog/static/6069361820131069485844/
1.自己寫一個類繼承自SqliteOpenHelper
2.會實現SqliteOpenHelper的兩個方法 onCreate與onUpgrade,google文檔對兩個回調方法的解釋是創建數據庫的時候調用與更新數據庫的版本的時候調用
3.Sqlite數據庫主要是用來緩存應用的數據,而應用卻是一直在更新版本,相應的數據的表的字段也會一直增加會改變或減少
4.這個時候就需要控制數據庫的版本,因爲Sqlite數據庫中的字段假設新版的應用裏面設計的表是10個字段,而緩存卻是之前緩存的只有9個字段的話,查詢數據庫之後的列
然後取的值會出現空指針異常或報錯
5.所以android中引入了Sqlite數據庫的版本,讓應用的舊版數據庫能夠與新版的數據庫的字段兼容
6.爲了兼容之前的數據庫的版本,只需要在應用的版本更新的時候,添加字段或者刪除字段即可
7.你開發程序當前是1.0.0的版本,該程序用到了數據庫,但是版本迭代之後到1.0.1的時候,數據庫的某個表添加了某個字段在軟件1.0.1的版本就需要升級
8.數據庫升級可以爲了能夠讓舊的數據不能丟,所以不能刪除掉之前數據庫中的所有數據,那麼就需要有地方能夠檢測到版本的變化,這個跟Android的APP升級是一個道理
當然這個檢測就是在SqliteOpenHelper的onUpgrade方法中
2. 數據庫升級應該注意什麼?
軟件的1.0版本升級到1.1版本時,老的數據不能丟。那麼在1.1版本的程序中就要有地方能夠檢測出來新的軟件版本與老的數據庫不兼容,並且能夠有辦法把1.0軟件的數據庫升級到1.1軟件能夠使用的數據庫。換句話說,要在1.0軟件的數據庫的那個表中增加那個字段,並賦予這個字段默認值。
3. 程序如何知道數據庫需要升級?
SQLiteOpenHelper類的構造函數有一個參數是int version,它的意思就是指數據庫版本號。比如在軟件1.0版本中,我們使用SQLiteOpenHelper訪問數據庫時,該參數爲1,那麼數據庫版本號1就會寫在我們的數據庫中。
到了1.1版本,我們的數據庫需要發生變化,那麼我們1.1版本的程序中就要使用一個大於1的整數來構造SQLiteOpenHelper類,用於訪問新的數據庫,比如2。
當我們的1.1新程序讀取1.0版本的老數據庫時,就發現老數據庫裏存儲的數據庫版本是1,而我們新程序訪問它時填的版本號爲2,系統就知道數據庫需要升級。
4. 何時觸發數據庫升級?如何升級?
當系統在構造SQLiteOpenHelper類的對象時,如果發現版本號不一樣,就會自動調用onUpgrade函數,讓你在這裏對數據庫進行升級。根據上述場景,在這個函數中把老版本數據庫的相應表中增加字段,並給每條記錄增加默認值即可。
新版本號和老版本號都會作爲onUpgrade函數的參數傳進來,便於開發者知道數據庫應該從哪個版本升級到哪個版本。
升級完成後,數據庫會自動存儲最新的版本號爲當前數據庫版本號。
做Android應用,不可避免的會與SQLite打交道。隨着應用的不斷升級,原有的數據庫結構可能已經不再適應新的功能,這時候,就需要對SQLite數據庫的結構進行升級了。 SQLite提供了ALTER TABLE命令,允許用戶重命名或添加新的字段到已有表中,但是不能從表中刪除字段。
並且只能在表的末尾添加字段,比如,爲 Subscription添加兩個字段:
1 ALTER TABLE Subscription ADD COLUMN Activation BLOB;
2 ALTER TABLE Subscription ADD COLUMN Key BLOB;
另外,如果遇到複雜的修改操作,比如在修改的同時,需要進行數據的轉移,那麼可以採取在一個事務中執行如下語句來實現修改表的需求。
1. 將表名改爲臨時表
ALTER TABLE Subscription RENAME TO __temp__Subscription;
2. 創建新表
CREATE TABLE Subscription (OrderId VARCHAR(32) PRIMARY KEY ,UserName VARCHAR(32) NOT NULL ,ProductId VARCHAR(16) NOT NULL);
3. 導入數據
INSERT INTO Subscription SELECT OrderId, “”, ProductId FROM __temp__Subscription;
或者
INSERT INTO Subscription() SELECT OrderId, “”, ProductId FROM __temp__Subscription;
* 注意 雙引號”” 是用來補充原來不存在的數據的
4. 刪除臨時表
DROP TABLE __temp__Subscription;
通過以上四個步驟,就可以完成舊數據庫結構向新數據庫結構的遷移,並且其中還可以保證數據不會應爲升級而流失。當然,如果遇到減少字段的情況,也可以通過創建臨時表的方式來實現。
Android應用程序更新的時候如果數據庫修改了字段需要更新數據庫,並且保留原來的數據庫數據:
這是原有的數據庫表
CREATE_BOOK = "create table book(bookId integer primarykey,bookName text);";
然後我們增加一個字段:
CREATE_BOOK = "create table book(bookId integer primarykey,bookName text,bookContent text);";
首先我們需要把原來的數據庫表重命名一下
CREATE_TEMP_BOOK = "alter table book rename to _temp_book";
然後把備份表中的數據copy到新創建的數據庫表中
INSERT_DATA = "insert into book select *,' ' from _temp_book";(注意' '是爲新加的字段插入默認值的必須加上,否則就會出錯)。
然後我們把備份表幹掉就行啦。
DROP_BOOK = "drop table _temp_book";
然後把數據庫的版本後修改一下,再次創建數據庫操作對象的時候就會自動更新(注:更新的時候第一個創建的操作數據的對象必須是可寫的,也就是通過這個方法getWritableDatabase()獲取的數據庫操作對象)
然後在onUpgrade()方法中執行上述sql語句就OK了
public class DBservice extends SQLiteOpenHelper{
private String CREATE_BOOK = "create table book(bookId integer primarykey,bookName text);";
private String CREATE_TEMP_BOOK = "alter table book rename to _temp_book";
private String INSERT_DATA = "insert into book select *,'' from _temp_book";
private String DROP_BOOK = "drop table _temp_book";
public DBservice(Context context, String name, CursorFactory factory,int version) {
super(context, name, factory, version);
}
@Override
public void onCreate(SQLiteDatabase db) {
db.execSQL(CREATE_BOOK);
}
@Override
public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {
switch (newVersion) {
case 2:
db.execSQL(CREATE_TEMP_BOOK);
db.execSQL(CREATE_BOOK);
db.execSQL(INSERT_DATA);
db.execSQL(DROP_BOOK);
break;
}
}