android 數據庫升級


原文地址:http://87426628.blog.163.com/blog/static/6069361820131069485844/


Andoird的SQLiteOpenHelper類中有一個onUpgrade方法。幫助文檔中只是說當數據庫升級時該方法被觸發。經過實踐,解決了我一連串的疑問:
1. 幫助文檔裏說的“數據庫升級”是指什麼?


你開發了一個程序,當前是1.0版本。該程序用到了數據庫。到1.1版本時,你在數據庫的某個表中增加了一個字段。那麼軟件1.0版本用的數據庫在軟件1.1版本就要被升級了。


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

 }


同類型好文:http://blog.sina.com.cn/s/blog_6400e5c50101mgu4.html

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章