Android教程網
  1. 首頁
  2. Android 技術
  3. Android 手機
  4. Android 系統教程
  5. Android 游戲
 Android教程網 >> Android技術 >> 關於Android編程 >> 四種Android數據存儲方式

四種Android數據存儲方式

編輯:關於Android編程

Android提供以下四種存儲方式

  • SharePreference
  • SQLite
  • File
  • ContentProvider

Android系統中數據基本都是私有的,一般存放在“data/data/程序包名”目錄下。如果要實現數據共享,正確的方式是使用ContentProvider。 

SharedPreference
SharedPreference是一種輕型的數據存儲方式,實際上是基於XML文件存儲的“key-value”鍵值對數據。通常用來存儲程序的一些配置信息。其存儲在“data/data/程序包名/shared_prefs目錄下。
SharedPreference本身只能獲取數據,不支持存儲和修改。存儲和修改要通過Editor對象來實現。 

1)、修改和存儲數據

  1. 根據Context的getSharedPrerences(key, [模式])方法獲取SharedPreference對象;
  2. 利用SharedPreference的editor()方法獲取Editor對象;
  3. 通過Editor的putXXX()方法,將鍵值對存儲數據;
  4. 通過Editor的commit()方法將數據提交到SharedPreference內。

綜合例子:   

//設置單例裡面的數值,然後再將數值寫入到SharedPreference裡

 private String setCityName(String _cityName){
  City.getCity().setCityName(_cityName);
  
  Context ctx =MainActivity.this;
  SharedPreferences sp =ctx.getSharedPreferences("CITY", MODE_PRIVATE);
  Editor editor=sp.edit();
  editor.putString("CityName", City.getCity().getCityName());
  editor.commit();
  
  return City.getCity().getCityName();
 }

2)、獲取數據

  1. 同樣根據Context對象獲取SharedPreference對象;
  2. 直接使用SharedPreference的getXXX(key)方法獲取數據。 

綜合例子:  

 //從單例裡面找,如果不存在則在SharedPreferences裡面讀取

 private String getCityName(){
  String cityName = City.getCity().getCityName();
  if(cityName==null ||cityName==""){
   Context ctx =MainActivity.this;
   SharedPreferences sp =ctx.getSharedPreferences("CITY", MODE_PRIVATE);
   City.getCity().setCityName(sp.getString("CityName", "廣州"));
  }
  return City.getCity().getCityName();
 }

注意
getSharedPrerences(key, [模式])方法中,第一個參數其實對應到XML的文件名,相同key的數據會保存到同一個文件下。
使用SharedPreference的getXXX(key)方法獲取數據的時候,如果key不存在的活,不會出現報錯,會返回none。建議使用getXXX()的時候指定默認值。

SQLite
SQLite是一個輕量級關系型數據庫,既然是關系型數據庫,那操作起來其實跟mysql、sql server差不多的。
需要注意的一點是,SQLite只有NULL、INTEGER、REAL(浮點數)、TEXT(字符串)和BLOB(大數據)五種類型,不存在BOOLEAN和DATE類型。 

1)、創建數據庫
通過openOrCreateDatabase(String path, SQLiteDatabase.CursorFactory factory)方法創建,如果庫已創建,則打開數據庫。
復制代碼 代碼如下:SQLiteDatabase db =this.openOrCreateDatabase("test_db.db", Context.MODE_PRIVATE, null);
2)、創建表
SQLiteDatabase沒有提供創建表的方法,所以要靠execSQL()方法來實現。看名字也知道execSQL()用於直接執行sql的。
復制代碼 代碼如下:String sql="create table t_user (id INTEGER PRIMARY KEY AUTOINCREMENT, name TEXT NOT NULL,password TEXT NOT NULL)";
db.execSQL(sql); 


使用SQLiteDatabase的insert(String table, String nullColumnHack, ContentValues values)方法插入數據。ContentValues 類,類似於java中的Map,以鍵值對的方式保存數據。

ContentValues values=new ContentValues();
values.put("name", "liangjh");
values.put("password", "123456");
db.insert("t_user", "id", values);


刪除數據就比較直接了。使用SQLiteDatabase的delete(String table, String whereClause, String[] whereArgs)實現。如果不想把參數寫在whereArgs裡面,可以直接把條件寫在whereClause裡面。

// 方式1 直接將條件寫入到條件裡面(個人覺得容易被注入,但其實數據都在客戶端,沒啥安全性可言)
db.delete("t_user", "id=1", null);
// 方式2 條件分開寫,感覺比較安全
db.delete("t_user", "name=? and password =?", new String[]{"weiyg","112233"});


        查詢有2個方法,query()和rawQuery()兩個方法,區別在於query()是將sql裡面的各參數提取出query()對應的參數中。可參考下面例子。

// 使用rawQuery
// Cursor c = db.rawQuery("select * from t_user", null);
// db.rawQuery("select * from t_user where id=1", null);
// db.rawQuery("select * from t_user where id=?", new String[]{"1"});
 
// 使用query()
Cursor c = db.query("t_user", new String[]{"id","name"}, "name=?", new String[]{"weiyg"}, null, null, null);
c.moveToFirst();
while(!c.isAfterLast()){
 String msg="";
 for(int i=0,j=c.getColumnCount();i<j;i++){
  msg+="--"+c.getString(i);
 }
 Log.v("SQLite", "data:"+msg);
 c.moveToNext();
}


        使用SQLiteDatabase的update(String table, ContentValues values, String whereClause, String[] whereArgs)可以修改數據。whereClause和whereArgs用於設置其條件。ContentValues對象為數據。

ContentValues values=new ContentValues();
values.put("password", "111111");
// 方式1 條件寫在字符串內
db.update("t_user", values, "id=1", null);
// 方式2 條件和字符串分開
db.update("t_user", values, "name=? or password=?",new String[]{"weiyg","123456"});

其它
無論何時,打開的數據庫,記得關閉。

db.close()
另外使用beginTransaction()和endTransaction()可以設置事務。 

File
        文件儲存方式,很久以前講過,這裡不說明。

ContentProvider
ContentProvider相對於其它的方式比較復雜,當然其功能相對於其它的方式也是革命性的改變。它能夠實現跨應用之間的數據操作。利用ContentResolver對象的delete、update、insert、query等方法去操ContentProvider的對象,讓ContentProvider對象的方法去對數據操作。實現方式為:

在A程序中定義一個ContentProvider,重載其增刪查改等方法;
在A程序中的AndroidManifest.xml中注冊ContentProvider;
在B程序中通過ContentResolver和Uri來獲取ContentProvider的數據,同樣利用Resolver的增刪查改方法來獲得和處理數據。
1)、在A程序定義一個Provider
新建一個類,繼承ContentProvider,並重載其delete()、insert()、query()、update()、getType()、onCreate()方法。譬如下面的例子,重載其onCreate和query方法。

public class MyProvider extends ContentProvider {

 @Override
 public int delete(Uri uri, String selection, String[] selectionArgs) {
  // TODO Auto-generated method stub
  return 0;
 }

 @Override
 public String getType(Uri uri) {
  // TODO Auto-generated method stub
  return null;
 }

 @Override
 public Uri insert(Uri uri, ContentValues values) {
  // TODO Auto-generated method stub
  return null;
 }

 @Override
 public boolean onCreate() {
  // 新建個數據庫並插入一條數據
  SQLiteDatabase db=this.getContext().openOrCreateDatabase("test_db2.db", Context.MODE_PRIVATE, null);
  db.execSQL("CREATE TABLE t_user (id INTEGER PRIMARY KEY AUTOINCREMENT,name TEXT NOT NULL)");
  ContentValues values=new ContentValues();
  values.put("name", "liangjh2");
  db.insert("t_user", "id", values);
  db.close();
  return false;
 }

 @Override
 public Cursor query(Uri uri, String[] projection, String selection,
   String[] selectionArgs, String sortOrder) {
  // 獲取數據
  SQLiteDatabase db=this.getContext().openOrCreateDatabase("test_db2.db", Context.MODE_PRIVATE, null);
  Cursor c = db.query("t_user", null, null, null, null, null, null);
  db.close();
  return c;
 }

 @Override
 public int update(Uri uri, ContentValues values, String selection,
   String[] selectionArgs) {
  // TODO Auto-generated method stub
  return 0;
 }

}

注冊ContentProvider
在AndroidManifest.xml中聲明ContentProvider,authorities屬性定義了ContentProvider的Uri標識。關於Uri標識屬另一個范疇,自行查詢。provider標識要放在<application></application>裡面。如果遇到了"Permission Denial: opening provide..."的錯誤,可以試試在節點加“android:exported="true"”。

<application ...>
 ...
 <provider android:name=".MyProvider" android:authorities="com.example.androidtestdemo" android:exported="true"/>
</application>

2)、在B程序獲取數據
用Context獲取到當前的ContentResolver,根據Uri地址和ContentResolver的query方法獲取A程序的數據。Uri地址和A程序中AndroidManifest.xml定義的autorities要一致。當然,同類可以進行其它的操作。

Context ctx=MainActivity.this;
ContentResolver resolver =ctx.getContentResolver();
Uri uri=Uri.parse("content://com.example.androidtestdemo");
Cursor c = resolver.query(uri, null, null, null, null);
c.moveToFirst();
while(!c.isAfterLast()){
 for(int i=0,j=c.getColumnCount();i<j;i++){
  Log.v("Android2",""+c.getString(i));
 }
 c.moveToNext();
}

以上就是本文的全部內容,希望對大家的學習有所幫助。

  1. 上一頁:
  2. 下一頁:
熱門文章
閱讀排行版
Copyright © Android教程網 All Rights Reserved