Android系統(tǒng)的五種數(shù)據(jù)存儲形式實(shí)例(一)
Android系統(tǒng)有五種數(shù)據(jù)存儲形式,分別是文件存儲、SP存儲、數(shù)據(jù)庫存儲、contentprovider 內(nèi)容提供者、網(wǎng)絡(luò)存儲。其中,前四個是本地存儲。存儲的類型包括簡單文本、窗口狀態(tài)存儲、音頻視頻數(shù)據(jù)、XML注冊文件的各種數(shù)據(jù)。各種存儲形式的特點(diǎn)不盡相同,因此對于不同的數(shù)據(jù)類型有著固定的存儲形式,本文為演示方便給出的案例基本相同,都是是采用賬號登錄來演示數(shù)據(jù)存儲,保存賬號和密碼信息,下次登錄時記住賬號和密碼。重在說明各種存儲形式的原理。
文件存儲:
以I/O流的形式把數(shù)據(jù)存入手機(jī)內(nèi)存或SD卡,可以存儲大數(shù)據(jù),如音樂、圖片或視頻等。對于手機(jī)內(nèi)存來說系統(tǒng)會根據(jù)每個應(yīng)用的包名創(chuàng)建一個/data/data/包名/的文件夾,訪問自己包名下的目錄是不需要權(quán)限的,并且 Android 已經(jīng)提供了非常簡便的 API 可以直接去訪問該文件夾。訪問時可以用getFilesDir()和getCacheDir(),兩個的區(qū)別是系統(tǒng)會自動清理后者中的內(nèi)容。
SD卡中的文件通常位于mnt/sdcard目錄下,不同生產(chǎn)商生產(chǎn)的手機(jī)這個路徑可能不同。操作sd卡的時通常要判斷下sd卡是否可用以及剩余空間是否足夠,因為部分手機(jī)的SD卡可卸載,SD卡處于非掛載狀態(tài)時,無法進(jìn)行讀寫操作。另外一點(diǎn),對SD卡的讀取和寫入操作均需要相應(yīng)的權(quán)限,否則無法完成。獲取SD卡路徑的方法是Environment.getExternalStorageDirectory(),其余操作與文件存儲基本類似。
文件存儲位置:
SD卡存儲路徑:
數(shù)據(jù)存儲在手機(jī)內(nèi)存的實(shí)現(xiàn)方法:
package com.example.qqload; import java.io.BufferedReader; import java.io.File; import java.io.FileInputStream; import java.io.FileNotFoundException; import java.io.FileOutputStream; import java.io.InputStreamReader; import com.example.qqload_sp.R; import android.os.Bundle; import android.app.Activity; import android.content.SharedPreferences; import android.content.SharedPreferences.Editor; import android.text.TextUtils; import android.view.Menu; import android.view.TextureView; import android.view.View; import android.widget.Button; import android.widget.CheckBox; import android.widget.EditText; import android.widget.Toast; public class MainActivity extends Activity { private EditText et_qq; private EditText et_password; private CheckBox cb_remenber; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); et_qq = (EditText) findViewById(R.id.et_qq); et_password = (EditText) findViewById(R.id.et_password); cb_remenber = (CheckBox) findViewById(R.id.cb_remenber); File file = new File(getFilesDir(), "info.txt"); // File file = new File(getCacheDir(), "info.txt"); 緩存中存放數(shù)據(jù) if (file.exists() && file.length() > 0) { try { FileInputStream fis = new FileInputStream(file); BufferedReader br = new BufferedReader(new InputStreamReader( fis)); String line = br.readLine(); String qq = line.split("##")[0]; String password = line.split("##")[1]; et_qq.setText(qq); et_password.setText(password); fis.close(); } catch (Exception e) { // TODO Auto-generated catch block e.printStackTrace(); } } } public void login(View view) { String qq = et_qq.getText().toString().trim(); String password = et_password.getText().toString().trim(); if (TextUtils.isEmpty(qq) || TextUtils.isEmpty(password)) { Toast.makeText(this, "密碼或者用戶名不能為空", 0).show(); return; } if (cb_remenber.isChecked()) { File file = new File(getFilesDir(), "info.txt"); try { FileOutputStream fos = new FileOutputStream(file); fos.write((qq + "##" + password).getBytes()); fos.close(); Toast.makeText(MainActivity.this, "保存成功", 0).show(); } catch (Exception e) { Toast.makeText(MainActivity.this, "保存失敗", 0).show(); e.printStackTrace(); } } } }
數(shù)據(jù)存儲在SD卡中的實(shí)現(xiàn)方法:
package com.example.qqload; import java.io.BufferedReader; import java.io.File; import java.io.FileInputStream; import java.io.FileNotFoundException; import java.io.FileOutputStream; import java.io.InputStreamReader; import java.text.Format; import com.example.qqload_sp.R; import android.os.Bundle; import android.os.Environment; import android.app.Activity; import android.content.SharedPreferences; import android.content.SharedPreferences.Editor; import android.text.TextUtils; import android.text.format.Formatter; import android.view.Menu; import android.view.TextureView; import android.view.View; import android.widget.Button; import android.widget.CheckBox; import android.widget.EditText; import android.widget.Toast; public class MainActivity extends Activity { private EditText et_qq; private EditText et_password; private CheckBox cb_remenber; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); et_qq = (EditText) findViewById(R.id.et_qq); et_password = (EditText) findViewById(R.id.et_password); cb_remenber = (CheckBox) findViewById(R.id.cb_remenber); File file = new File(Environment.getExternalStorageDirectory(), "info.txt"); if (file.exists() && file.length() > 0) { try { FileInputStream fis = new FileInputStream(file); BufferedReader br = new BufferedReader(new InputStreamReader( fis)); String line = br.readLine(); String qq = line.split("##")[0]; String password = line.split("##")[1]; et_qq.setText(qq); et_password.setText(password); fis.close(); } catch (Exception e) { // TODO Auto-generated catch block e.printStackTrace(); } } } public void login(View view) { String qq = et_qq.getText().toString().trim(); String password = et_password.getText().toString().trim(); if (TextUtils.isEmpty(qq) || TextUtils.isEmpty(password)) { Toast.makeText(this, "密碼或者用戶名不能為空", 0).show(); return; } if (cb_remenber.isChecked()) { File file = new File(Environment.getExternalStorageDirectory(), "info.txt"); //判斷SD卡是否掛載 if(!Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED)){ Toast.makeText(MainActivity.this, "SD卡不可用", 0).show(); return; } //判斷SD卡大小是否充足 long size = Environment.getExternalStorageDirectory().getFreeSpace(); String info = Formatter.formatFileSize(this, size); //此處存儲數(shù)據(jù)較小就不進(jìn)行判斷 Toast.makeText(this, "可用空間" + info, 0).show(); try { FileOutputStream fos = new FileOutputStream(file); fos.write((qq + "##" + password).getBytes()); fos.close(); Toast.makeText(MainActivity.this, "保存成功", 0).show(); } catch (Exception e) { Toast.makeText(MainActivity.this, "保存失敗", 0).show(); e.printStackTrace(); } } } }
SP存儲:
SP存儲本質(zhì)上是一個XML文件,以鍵值對的形式存入手機(jī)內(nèi)存中。常用于存儲簡單的參數(shù)設(shè)置,如登陸賬號密碼的存儲,窗口功能狀態(tài)的存儲等,該存儲文件位于:data/data/包名/shared_prefs文件夾中。使用的時候,首先需要通過context.getSharedPrefrences(String name,int mode)獲取SharedPrefrences的實(shí)例對象,存儲數(shù)據(jù)時,用SharedPrefrences的實(shí)例對象得到SharedPrefrences文件的編輯器,在編輯器中用putXxx()添加數(shù)據(jù),之后務(wù)必用commit提交數(shù)據(jù),否則無法獲取數(shù)據(jù)。取數(shù)據(jù)時,直接用getXxx()方法。
sp存儲自動生成xml文件,其的路徑如下:
sp存儲的實(shí)現(xiàn)方法:
package com.example.qqload; import com.example.qqload_sp.R; import android.os.Bundle; import android.app.Activity; import android.content.SharedPreferences; import android.content.SharedPreferences.Editor; import android.text.TextUtils; import android.view.Menu; import android.view.TextureView; import android.view.View; import android.widget.Button; import android.widget.CheckBox; import android.widget.EditText; import android.widget.Toast; public class MainActivity extends Activity { private EditText et_qq; private EditText et_password; private CheckBox cb_remenber; private SharedPreferences sp; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); et_qq = (EditText) findViewById(R.id.et_qq); et_password = (EditText) findViewById(R.id.et_password); cb_remenber = (CheckBox) findViewById(R.id.cb_remenber); sp = this.getSharedPreferences("config", 0); String qq = sp.getString("qq",""); String password = sp.getString("password",""); et_qq.setText(qq); et_password.setText(password); } public void login(View view){ String qq = et_qq.getText().toString().trim(); String password = et_password.getText().toString().trim(); if(TextUtils.isEmpty(qq)||TextUtils.isEmpty(password)){ Toast.makeText(this,"密碼或者用戶名不能為空",0).show(); return; } if(cb_remenber.isChecked()){ Editor edit = sp.edit(); edit.putString("qq",qq); edit.putString("password",password); edit.commit(); } } }
數(shù)據(jù)庫存儲:
數(shù)據(jù)庫所有信息都存儲在單一文件內(nèi),占用內(nèi)存小,并且支持基本SQL語法,是項目中經(jīng)常被采用的一種數(shù)據(jù)存儲方式,通常用于存儲用戶信息等,例如在手機(jī)上做一個學(xué)生信息管理系統(tǒng)。SQLite 是一款內(nèi)置到移動設(shè)備上的輕量型的數(shù)據(jù)庫,SQLiteOpenHelper 是Android 提供的一個抽象工具類,負(fù)責(zé)管理數(shù)據(jù)庫的創(chuàng)建、升級工作。數(shù)據(jù)庫的路徑為:/data/data/應(yīng)用包名/databases/數(shù)據(jù)庫。如果想創(chuàng)建數(shù)據(jù)庫,就需要自定義一個類繼承SQLiteOpenHelper,然后覆寫其中的抽象方法,指定數(shù)據(jù)庫名、版本號。在onCreate() 方法中通過執(zhí)行sql 語句實(shí)現(xiàn)表的創(chuàng)建。如果只是創(chuàng)建出來該類并不會真正的去創(chuàng)建數(shù)據(jù)庫,而是需要通過執(zhí)行helper.getWritableDatabase()或者h(yuǎn)epler.getReadableDatabase()。另外想要對創(chuàng)建的數(shù)據(jù)庫進(jìn)行增刪改查的操作可以單獨(dú)定義一個類實(shí)現(xiàn)。增刪改查操作有兩種方式,一是直接執(zhí)行sql語句,另一個是Android自身的API實(shí)現(xiàn)。用數(shù)據(jù)庫實(shí)現(xiàn)賬號登錄顯得有些大材小用,為演示數(shù)據(jù)庫的原理本文給出的案例是用數(shù)據(jù)庫記錄多個用戶的賬號和密碼信息,并把最后一個賬號信息回顯在界面。但實(shí)際應(yīng)用中很少這樣做。
從手機(jī)文件中導(dǎo)出數(shù)據(jù)庫文件并不可以直接打開,因此可以用可視化工具和sqlite3操作工具進(jìn)行查看。這里介紹sqlite3工具的使用。具體需要的步驟如下:
1. 執(zhí)行adb shell命令進(jìn)入Linuxne內(nèi)核;
2. 使用cd進(jìn)入數(shù)據(jù)庫所在的路徑 cd: /data/data/應(yīng)用包名/databases;
3. 進(jìn)入數(shù)據(jù)庫模式: sqlite3 數(shù)據(jù)庫名.db;
4. 執(zhí)行SQL語句
Sqlite3操作演示:
數(shù)據(jù)庫存儲路徑:
數(shù)據(jù)庫實(shí)現(xiàn)方法,先創(chuàng)建一個類繼承SqliteOpenHelper,在類中創(chuàng)建數(shù)據(jù)庫和表:
package com.example.qqload.db; import android.content.Context; import android.database.sqlite.SQLiteDatabase; import android.database.sqlite.SQLiteDatabase.CursorFactory; import android.database.sqlite.SQLiteOpenHelper; public class UserDBOpenhelper extends SQLiteOpenHelper { public UserDBOpenhelper(Context context) { super(context, "user.db", null, 1); // TODO Auto-generated constructor stub } @Override public void onCreate(SQLiteDatabase arg0) { arg0.execSQL("create table user (_id integer primary key autoincrement,name vachar(20),password varchar(20))"); } @Override public void onUpgrade(SQLiteDatabase arg0, int arg1, int arg2) { // TODO Auto-generated method stub } }
創(chuàng)建一個工具類實(shí)現(xiàn)對數(shù)據(jù)庫的操作:
package com.example.qqload.db.dao; import java.util.ArrayList; import java.util.List; import android.content.ContentValues; import android.content.Context; import android.database.Cursor; import android.database.sqlite.SQLiteDatabase; import com.example.qqload.db.UserDBOpenhelper; public class UserDao { private UserDBOpenhelper helper; public UserDao(Context context){ helper = new UserDBOpenhelper(context); } public long add(String name,String password){ SQLiteDatabase db = helper.getWritableDatabase(); //用SQL語句實(shí)現(xiàn)增加數(shù)據(jù)的功能 //db.execSQL("insert into user (name,passeord) values (?,?)", new Object[]{name,password}); //android自身API實(shí)現(xiàn)修改功能可以有返回值 ContentValues values =new ContentValues(); values.put("name", name); values.put("password", password); long result = db.insert("user", null, values); //帶返回值,表示添加在哪一行 db.close(); return result; } public List<user> findAll(){ List<user> list =new ArrayList<user>(); SQLiteDatabase db = helper.getReadableDatabase(); //Cursor cursor = db.rawQuery("select name, password from user", null); Cursor cursor = db.query("user", new String[]{"name","password"}, null, null, null, null, null); while(cursor.moveToNext()){ String name = cursor.getString(0); String password = cursor.getString(1); user us = new user(); us.setName(name); us.setPassword(password); list.add(us); } cursor.close(); db.close(); return list; } }
在主方法中實(shí)現(xiàn)賬號登錄和記錄
package com.example.qqload; import java.io.BufferedReader; import java.io.File; import java.io.FileInputStream; import java.io.FileNotFoundException; import java.io.FileOutputStream; import java.io.InputStreamReader; import java.util.List; import com.example.qqload.db.dao.UserDao; import com.example.qqload.db.dao.user; import com.example.qqload.R; import android.os.Bundle; import android.app.Activity; import android.content.SharedPreferences; import android.content.SharedPreferences.Editor; import android.text.TextUtils; import android.view.Menu; import android.view.TextureView; import android.view.View; import android.widget.Button; import android.widget.CheckBox; import android.widget.EditText; import android.widget.Toast; public class MainActivity extends Activity { private EditText et_qq; private EditText et_password; private CheckBox cb_remenber; private UserDao dao; private List<user> list; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); et_qq = (EditText) findViewById(R.id.et_qq); et_password = (EditText) findViewById(R.id.et_password); cb_remenber = (CheckBox) findViewById(R.id.cb_remenber); user u = new user(); dao = new UserDao(MainActivity.this); list = dao.findAll(); if (list.size() == 0) { et_qq.setText(""); et_password.setText(""); } else { System.out.println("大?。? + list.size()); for (int i = 0; i < list.size(); i++) { System.out.println(list.get(i).getName() + "::::" + list.get(i).getPassword()); } u = list.get(list.size()-1); String qq = u.getName(); String password = u.getPassword(); et_qq.setText(qq); et_password.setText(password); } } public void login(View view) { String qq = et_qq.getText().toString().trim(); String password = et_password.getText().toString().trim(); if (TextUtils.isEmpty(qq) || TextUtils.isEmpty(password)) { Toast.makeText(this, "密碼或者用戶名不能為空", 0).show(); return; } if (cb_remenber.isChecked()) { dao.add(qq, password); //在工具類添加增加功能 Toast.makeText(MainActivity.this, "保存成功", 0).show(); } } }
用數(shù)據(jù)庫實(shí)現(xiàn)賬號登錄案例的目錄結(jié)構(gòu)如下所示:
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Android 將網(wǎng)絡(luò)的Url資源轉(zhuǎn)換為Drawable資源方式
這篇文章主要介紹了Android 將網(wǎng)絡(luò)的Url資源轉(zhuǎn)換為Drawable資源方式,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2020-03-03Android中LeakCanary檢測內(nèi)存泄漏的方法
本篇文章主要介紹了Android中LeakCanary檢測內(nèi)存泄漏的方法,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2017-09-09Android SQLite數(shù)據(jù)庫進(jìn)行查詢優(yōu)化的方法
這篇文章主要給大家介紹了關(guān)于Android SQLite數(shù)據(jù)庫進(jìn)行查詢優(yōu)化的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2018-11-11Android studio利用gradle打jar包并混淆的方法詳解
昨天準(zhǔn)備把寫好的代碼使用gradle打jar包出來,并打算加混淆。打jar包容易,結(jié)果在混淆上走了彎路。所以這篇文章主要介紹了關(guān)于Android studio利用gradle打jar包并混淆的方法,需要的朋友可以參考下。2017-03-03Android 中對于圖片的內(nèi)存優(yōu)化方法
Android 中對于圖片的內(nèi)存優(yōu)化方法,需要的朋友可以參考一下2013-03-03Android在項目中接入騰訊TBS瀏覽器WebView的教程與注意的地方
今天小編就為大家分享一篇關(guān)于Android在項目中接入騰訊TBS瀏覽器WebView的教程與注意的地方,小編覺得內(nèi)容挺不錯的,現(xiàn)在分享給大家,具有很好的參考價值,需要的朋友一起跟隨小編來看看吧2018-10-10Android組件化開發(fā)路由的設(shè)計實(shí)踐
本篇文章主要介紹了Android組件化開發(fā)路由的設(shè)計實(shí)踐,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2018-05-05Android 監(jiān)聽屏幕是否鎖屏的實(shí)例代碼
今天小編通過本文給大家分享android如何監(jiān)聽手機(jī)屏幕是否鎖屏。實(shí)現(xiàn)方法很簡單,需要的朋友參考下吧2017-09-09