Android開發(fā)筆記之Android中數(shù)據(jù)的存儲方式(一)
對于開發(fā)平臺來講,如果對數(shù)據(jù)的存儲有良好的支持,那么對應用程序的開發(fā)將會有很大的促進作用。
總體的來講,數(shù)據(jù)存儲方式有三種:一個是文件,一個是數(shù)據(jù)庫,另一個則是網(wǎng)絡。其中文件和數(shù)據(jù)庫可能用的稍多一些,文件用起來較為方便,程序可以自己定義格式;數(shù)據(jù)庫用起稍煩鎖一些,但它有它的優(yōu)點,比如在海量數(shù)據(jù)時性能優(yōu)越,有查詢功能,可以加密,可以加鎖,可以跨應用,跨平臺等等;網(wǎng)絡,則用于比較重要的事情,比如科研,勘探,航空等實時采集到的數(shù)據(jù)需要馬上通過網(wǎng)絡傳輸?shù)綌?shù)據(jù)處理中心進行存儲并進行處理,有實時性的需求等。
對于Android平臺來講,它的存儲方式也不外乎這幾種,按方式總體來分,也是文件,數(shù)據(jù)庫和網(wǎng)絡。但我認為從儲存標的來講它可以細分為以下五種方式:
1.1 方式
1.Shared Preferences:主要用于保存程序的系統(tǒng)配置信息。用來存儲“key-values paires”。一般用于保存程序啟動時設定的信息,以便在程序下一次啟動時繼續(xù)保留前一次設定的信息。
2.xml:保存復雜數(shù)據(jù),比如短信備份。
3.Files:用文件的形式保存信息。可以通過對文件的讀寫來獲取或保存相關信息。
4.SQLite:用數(shù)據(jù)庫的形式保存信息。SQLite是一個開源的數(shù)據(jù)庫 系統(tǒng)。
5.NetWork:將數(shù)據(jù)保存于網(wǎng)絡。
1.2 區(qū)別
1. Shared Preferences:
Android提供用來存儲一些簡單的配置信息的一種機制,例如,一些默認歡迎語、登錄的用戶名和密碼等。其以鍵值對的方式存儲。
SharedPreferences是以XML的格式以文件的方式自動保存的,在DDMS中的File Explorer中展開到/data/data/<packagename>/shared_prefs下,以自己的項目為例,可以看到一個叫做SETTING_Infos.xml的文件
2. Files
在Android中,其提供了openFileInput 和 openFileOuput 方法讀取設備上的文件,下面看個例子代碼,具體如下所示:
String FILE_NAME = "tempfile.tmp"; //確定要操作文件的文件名 FileOutputStream fos = openFileOutput(FILE_NAME, Context.MODE_PRIVATE); //初始化 FileInputStream fis = openFileInput(FILE_NAME); //創(chuàng)建寫入流
上述代碼中兩個方法只支持讀取該應用目錄下的文件,讀取非其自身目錄下的文件將會拋出異常。需要提醒的是,如果調(diào)用FileOutputStream 時指定的文件不存在,Android 會自動創(chuàng)建它,所以不需要判斷文件是否存在。另外,在默認情況下,寫入的時候會覆蓋原文件內(nèi)容,如果想把新寫入的內(nèi)容附加到原文件內(nèi)容后,則可以指定其模式為Context.MODE_APPEND,這里涉及到openFileOutput()的四種模式,下文我會詳細解釋和案例。
3. SQLite
SQLite是Android所帶的一個標準的數(shù)據(jù)庫,它支持SQL語句,它是一個輕量級的嵌入式數(shù)據(jù)庫
4. NetWork:
將數(shù)據(jù)上傳到網(wǎng)絡
補充:
1.Shared Preferences底層使用xml,xml也可以保存數(shù)據(jù),但是Shared Preferences只能保存鍵值對方式,xml能保存復雜數(shù)據(jù)
2.Content provider底部還是使用了Sqlite數(shù)據(jù)庫,也是算一種方式。
1.3 例子
1. Shared Preferences:
小案例:用戶輸入賬號密碼,點擊登錄按鈕,登錄的同時持久化保存賬號和密碼
用SharedPreference存儲賬號密碼
•往SharedPreference里寫數(shù)據(jù)
//拿到一個SharedPreference對象 SharedPreferences sp = getSharedPreferences("config", MODE_PRIVATE); //拿到編輯器 Editor ed = sp.edit(); //寫數(shù)據(jù) ed.putString("name", name); ed.commit();
注:這里記住put完后,必須commit一下。
•從SharedPreference里取數(shù)據(jù)
SharedPreferences sp = getSharedPreferences("config", MODE_PRIVATE); //從SharedPreference里取數(shù)據(jù) String name = sp.getBoolean("name", "");
•代碼:
•布局文件
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_height="match_parent" android:paddingBottom="@dimen/activity_vertical_margin" android:paddingLeft="@dimen/activity_horizontal_margin" android:paddingRight="@dimen/activity_horizontal_margin" android:paddingTop="@dimen/activity_vertical_margin" tools:context=".MainActivity" android:orientation="vertical" > <EditText android:id="@+id/et_name" android:layout_width="match_parent" android:layout_height="wrap_content" android:hint="請輸入用戶名" /> <EditText android:id="@+id/et_pass" android:layout_width="match_parent" android:layout_height="wrap_content" android:hint="請輸入密碼" android:inputType="textPassword" /> <RelativeLayout android:layout_width="match_parent" android:layout_height="wrap_content" > <CheckBox android:id="@+id/cb" android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="記住用戶名和密碼" android:layout_centerVertical="true" /> <Button android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="登錄" android:layout_alignParentRight="true" android:onClick="login" /> </RelativeLayout> </LinearLayout>
•java代碼
package com.bokeyuan.sharedpreference; import android.os.Bundle; import android.app.Activity; import android.content.SharedPreferences; import android.content.SharedPreferences.Editor; import android.view.Menu; import android.view.View; import android.widget.EditText; public class MainActivity extends Activity { private EditText et_name; private EditText et_pass; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); et_name = (EditText) findViewById(R.id.et_name); et_pass = (EditText) findViewById(R.id.et_pass); //拿到SharedPreferences對象 SharedPreferences sp = getSharedPreferences("info", MODE_PRIVATE); //從SharedPreferences中取出數(shù)據(jù) String name = sp.getString("name", ""); String pass = sp.getString("pass", ""); et_name.setText(name); et_pass.setText(pass); } public void login(View v){ String name = et_name.getText().toString(); String pass = et_pass.getText().toString(); //拿到SharedPreferences對象 SharedPreferences sp = getSharedPreferences("info", MODE_PRIVATE); //把數(shù)據(jù)存入SharedPreferences Editor ed = sp.edit(); ed.putString("name", name); ed.putString("pass", pass); //提交 ed.commit(); } }
2. Files
小案例:用戶輸入賬號密碼,勾選“記住賬號密碼”,點擊登錄按鈕,登錄的同時持久化保存賬號和密碼
•在內(nèi)部存儲空間中讀寫文件(RAM)
•布局文件:同上面的布局
•java代碼:
package com.bokeyuan.rwinrom; 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 android.os.Bundle; import android.annotation.SuppressLint; import android.app.Activity; import android.content.Context; import android.view.Menu; import android.view.View; import android.widget.CheckBox; import android.widget.EditText; import android.widget.Toast; public class MainActivity extends Activity { private EditText et_name; private EditText et_pass; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); et_name = (EditText) findViewById(R.id.et_name); et_pass = (EditText) findViewById(R.id.et_pass); readAccount(); } @SuppressLint("ShowToast") public void login(View v){ String name = et_name.getText().toString(); String pass = et_pass.getText().toString(); CheckBox cb = (CheckBox) findViewById(R.id.cb); if(cb.isChecked()){ //指定Android的內(nèi)部存儲空間的路徑 // File file = new File("data/data/com.bokeyuan.rwinrom/info.txt"); //返回一個File對象,它的路徑是:data/data/com.bokeyuan.rwinrom/files/ // File file = new File(getFilesDir(), "info.txt"); //返回一個File對象,它的路徑是:data/data/com.bokeyuan.rwinrom/cache/ File file = new File(getCacheDir(), "info.txt"); try { FileOutputStream fos = new FileOutputStream(file); fos.write((name + "##" + pass).getBytes()); fos.close(); } catch (Exception e) { // TODO Auto-generated catch block e.printStackTrace(); } } //彈出提示框,提示用戶登錄成功 Toast.makeText(this, "登陸成功", 0).show(); } public void readAccount(){ //指定Android的內(nèi)部存儲空間的路徑 // File file = new File("data/data/com.bokeyuan.rwinrom/info.txt"); // File file = new File(getFilesDir(), "info.txt"); File file = new File(getCacheDir(), "info.txt"); if(file.exists()){ try { FileInputStream fis = new FileInputStream(file); //把字節(jié)流轉(zhuǎn)換成字符流 BufferedReader br = new BufferedReader(new InputStreamReader(fis)); String text = br.readLine(); String[] s = text.split("##"); et_name.setText(s[0]); et_pass.setText(s[1]); } catch (Exception e) { // TODO Auto-generated catch block e.printStackTrace(); } } } }
•在外部存儲空間中讀寫文件(SD卡)
•布局文件:同上面的布局
•java代碼:
package com.bokeyuan.rwinsd; 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.bokeyuan.rwinsd.R; import android.os.Bundle; import android.os.Environment; import android.annotation.SuppressLint; import android.app.Activity; import android.content.Context; import android.view.Menu; import android.view.View; import android.widget.CheckBox; import android.widget.EditText; import android.widget.Toast; public class MainActivity extends Activity { private EditText et_name; private EditText et_pass; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); et_name = (EditText) findViewById(R.id.et_name); et_pass = (EditText) findViewById(R.id.et_pass); readAccount(); } public void login(View v){ String name = et_name.getText().toString(); String pass = et_pass.getText().toString(); CheckBox cb = (CheckBox) findViewById(R.id.cb); if(cb.isChecked()){ //檢測sd卡當前是否可用 if(Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED)){ // File file = new File("sdcard/info.txt"); //返回一個file對象,包含的路徑就是sd卡的真實路徑 File file = new File(Environment.getExternalStorageDirectory(), "info.txt"); try { FileOutputStream fos = new FileOutputStream(file); fos.write((name + "##" + pass).getBytes()); fos.close(); } catch (Exception e) { // TODO Auto-generated catch block e.printStackTrace(); } } else{ Toast.makeText(this, "sd卡不可用喲親", 0).show(); } } //彈出提示框,提示用戶登錄成功 Toast.makeText(this, "登陸成功", 0).show(); } public void readAccount(){ // File file = new File("sdcard/info.txt"); File file = new File(Environment.getExternalStorageDirectory(), "info.txt"); if(file.exists()){ try { FileInputStream fis = new FileInputStream(file); //把字節(jié)流轉(zhuǎn)換成字符流 BufferedReader br = new BufferedReader(new InputStreamReader(fis)); String text = br.readLine(); String[] s = text.split("##"); et_name.setText(s[0]); et_pass.setText(s[1]); } catch (Exception e) { // TODO Auto-generated catch block e.printStackTrace(); } } } }
注:
•//此api會把文件寫到data/data/com.itheima.permission/files/文件夾下
•FileOutputStream fos = openFileOutput("info1.txt", MODE_PRIVATE) ;
•//此api會把文件讀到data/data/com.itheima.permission/files/文件夾下
•FileOutputStream fos = openFileOutput("info1.txt", MODE_PRIVATE) ;
•所以,以后用Android 自帶的API。可以看下面openFileOutput四種模式的小例子:
•openFileOutput的四種模式
•MODE_PRIVATE = 0: -rw-rw---- •MODE_APPEND = 32768: -rw-rw---- •MODEWORLDREADABLE = 1: -rw-rw-r-- •MODEWORLDWRITEABLE = 2: -rw-rw--w-
Context.MODE_PRIVATE:為默認操作模式,代表該文件是私有數(shù)據(jù),只能被應用本身訪問,在該模式下,寫入的內(nèi)容會覆蓋原文件的內(nèi)容,如果想把新寫入的內(nèi)容追加到原文件中,可以使用Context.MODE_APPEND。
Context.MODE_APPEND:模式會檢查文件是否存在,存在就往文件追加內(nèi)容,否則就創(chuàng)建新文件。
Context.MODE_WORLD_READABLE和Context.MODE_WORLD_WRITEABLE用來控制其他應用是否有權限讀寫該文件。
MODE_WORLD_READABLE:表示當前文件可以被其他應用讀??;MODE_WORLD_WRITEABLE:表示當前文件可以被其他應用寫入。
• 注:
•在Android中,每一個應用是一個獨立的用戶
•drwxrwxrwx
•第1位:d表示文件夾,-表示文件
•第2-4位:rwx,表示這個文件的擁有者用戶(owner)對該文件的權限 •r:讀
•w:寫
•x:執(zhí)行
•第5-7位:rwx,表示跟文件擁有者用戶同組的用戶(grouper)對該文件的權限
•第8-10位:rwx,表示其他用戶組的用戶(other)對該文件的權限
•布局文件:
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_height="match_parent" android:paddingBottom="@dimen/activity_vertical_margin" android:paddingLeft="@dimen/activity_horizontal_margin" android:paddingRight="@dimen/activity_horizontal_margin" android:paddingTop="@dimen/activity_vertical_margin" tools:context=".MainActivity" android:orientation="vertical" > <Button android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="創(chuàng)建文件1" android:onClick="click1" /> <Button android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="創(chuàng)建文件2" android:onClick="click2" /> <Button android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="創(chuàng)建文件3" android:onClick="click3" /> </LinearLayout>
•代碼:
package com.bokeyuan.permission; import java.io.File; import java.io.FileNotFoundException; import java.io.FileOutputStream; import android.os.Bundle; import android.annotation.SuppressLint; import android.app.Activity; import android.view.Menu; import android.view.View; @SuppressLint("WorldReadableFiles") public class MainActivity extends Activity { @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); } public void click1(View v){ //此api會把文件寫到data/data/com.bokeyuan.permission/files/文件夾下 try { FileOutputStream fos = openFileOutput("info1.txt", MODE_PRIVATE) ; fos.write("該文件是私有數(shù)據(jù),只能被應用本身訪問".getBytes()); fos.close(); } catch (Exception e) { // TODO Auto-generated catch block e.printStackTrace(); } } public void click2(View v){ //此api會把文件寫到data/data/com.bokeyuan.permission/files/文件夾下 try { @SuppressWarnings("deprecation") FileOutputStream fos = openFileOutput("info2.txt", MODE_WORLD_READABLE | MODE_WORLD_WRITEABLE) ; fos.write("當前文件可以被其他應用讀取或?qū)懭?.getBytes()); fos.close(); } catch (Exception e) { // TODO Auto-generated catch block e.printStackTrace(); } } public void click3(View v){ //此api會把文件寫到data/data/com.bokeyuan.permission/files/文件夾下 try { @SuppressWarnings("deprecation") FileOutputStream fos = openFileOutput("info3.txt", MODE_WORLD_WRITEABLE) ; fos.write("當前文件可以被其他應用寫入".getBytes()); fos.close(); } catch (Exception e) { // TODO Auto-generated catch block e.printStackTrace(); } } }
•獲取SD卡剩余容量:
•有時候讀寫文件的時候,需要判斷SD卡的剩余容量,再進行寫入操作。下面小例子,引用Android系統(tǒng)底層的API,獲取SD卡的剩余容量。
•布局文件:
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_height="match_parent" android:paddingBottom="@dimen/activity_vertical_margin" android:paddingLeft="@dimen/activity_horizontal_margin" android:paddingRight="@dimen/activity_horizontal_margin" android:paddingTop="@dimen/activity_vertical_margin" tools:context=".MainActivity" > <TextView android:id="@+id/tv" android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="@string/hello_world" /> </RelativeLayout>
•Java代碼:
package com.bokeyuan.getsdavail; import java.io.File; import android.os.Build; import android.os.Bundle; import android.os.Environment; import android.os.StatFs; import android.app.Activity; import android.text.format.Formatter; import android.view.Menu; import android.widget.TextView; public class MainActivity extends Activity { @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); File path = Environment.getExternalStorageDirectory(); StatFs stat = new StatFs(path.getPath()); long blockSize; long totalBlocks; long availableBlocks; //檢測系統(tǒng)當前版本號 if(Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN_MR2){ blockSize = stat.getBlockSizeLong(); totalBlocks = stat.getBlockCountLong(); availableBlocks = stat.getAvailableBlocksLong(); } else{ blockSize = stat.getBlockSize(); totalBlocks = stat.getBlockCount(); availableBlocks = stat.getAvailableBlocks(); } TextView tv = (TextView) findViewById(R.id.tv); tv.setText(formatSize(availableBlocks * blockSize)); } private String formatSize(long size) { return Formatter.formatFileSize(this, size); } }
以上所述給大家介紹了Android開發(fā)筆記之Android中數(shù)據(jù)的存儲方式(一)的相關知識,希望本文分享能夠幫助到大家。下篇給android開發(fā)筆記之a(chǎn)ndroid中數(shù)據(jù)的存儲方式(二),感興趣的朋友點擊了解詳情。
- Android sdcard實現(xiàn)圖片存儲 、聯(lián)網(wǎng)下載
- Android持久化技術之SharedPreferences存儲實例詳解
- 詳解Android文件存儲
- 實例詳解Android文件存儲數(shù)據(jù)方式
- Android開發(fā)筆記之Android中數(shù)據(jù)的存儲方式(二)
- Android數(shù)據(jù)存儲之SQLite使用
- Android編程使用內(nèi)容提供者方式(ContentProvider)進行存儲的方法
- 詳解Android四種存儲方式
- Android編程中的5種數(shù)據(jù)存儲方式
- Android使用文件進行數(shù)據(jù)存儲的方法
- Android編程之SharedPreferences文件存儲操作實例分析
- android中使用SharedPreferences進行數(shù)據(jù)存儲的操作方法
- Android應用開發(fā)SharedPreferences存儲數(shù)據(jù)的使用方法
- 簡介Android應用中sharedPreferences類存儲數(shù)據(jù)的用法
相關文章
android ViewPager實現(xiàn)自動無限輪播和下方向?qū)A點
本篇文章主要介紹了android ViewPager實現(xiàn)自動輪播和下方向?qū)A點,具有一定的參考價值,感興趣的小伙伴們可以參考一下。2017-02-02Android 8.1 Launcher3實現(xiàn)動態(tài)指針時鐘功能
這篇文章主要介紹了Android 8.1 Launcher3實現(xiàn)動態(tài)指針時鐘功能,本文通過實例代碼給大家介紹的非常詳細,具有一定的參考借鑒價值,需要的朋友可以參考下2018-07-07android之json數(shù)據(jù)過長打印不全問題的解決
這篇文章主要介紹了android之json數(shù)據(jù)過長打印不全問題的解決,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2020-04-04