Android数据存储之文件存储(瞬时数据的存储与读取)项目已上传GitHub
GitHub地址:
https://github.com/Skymqq/FileSave.git
?
數(shù)據(jù)持久化就是指將那些內(nèi)存中的瞬時(shí)數(shù)據(jù)保存到存儲(chǔ)設(shè)備中,保證即使在手機(jī)或電腦關(guān)機(jī)的情況下,這些數(shù)據(jù)仍然不會(huì)丟失。保存在內(nèi)存中的數(shù)據(jù)是處于瞬時(shí)狀態(tài)的,而保存在存儲(chǔ)設(shè)備中的數(shù)據(jù)是處于持久狀態(tài)的,持久化技術(shù)則提供了一種機(jī)制可以讓數(shù)據(jù)在瞬時(shí)狀態(tài)和持久狀態(tài)之間進(jìn)行轉(zhuǎn)換。
Android系統(tǒng)中主要提供了3中方式用于簡(jiǎn)單地實(shí)現(xiàn)數(shù)據(jù)持久化功能:
1.文件存儲(chǔ)
2.SharedPreferences存儲(chǔ)
3.數(shù)據(jù)庫(kù)存儲(chǔ)
除了以上3中方式之外,你還可以將數(shù)據(jù)保存在SD卡中,不過(guò)使用文件、SharedPreferences或數(shù)據(jù)庫(kù)來(lái)保存數(shù)據(jù)會(huì)相對(duì)簡(jiǎn)單一些,而且也比SD存儲(chǔ)更安全。
?
文件存儲(chǔ)
文件存儲(chǔ)是Android中最基本的一種數(shù)據(jù)存儲(chǔ)方式,它不對(duì)存儲(chǔ)的內(nèi)容進(jìn)行任何的格式化處理,所有的數(shù)據(jù)都是原封不動(dòng)地保存到文件當(dāng)中的,因而它比較適合存儲(chǔ)一些簡(jiǎn)單的文本數(shù)據(jù)或二進(jìn)制數(shù)據(jù)。如果你想使用文件存儲(chǔ)的方式來(lái)保存一些較為復(fù)雜的文本數(shù)據(jù),就需要定義一套自己的格式規(guī)范,這樣可以方便之后將數(shù)據(jù)從文件中重新解析出來(lái)。
那么我們就來(lái)看一下,Android中是如何通過(guò)文件來(lái)保存數(shù)據(jù)的。
將數(shù)據(jù)存儲(chǔ)到文件中:
Context類中提供了一個(gè)openFileOutput()方法,可以用于將數(shù)據(jù)存儲(chǔ)到指定的文件中。這個(gè)方法接收兩個(gè)參數(shù),第一個(gè)是文件名,在文件創(chuàng)建的時(shí)候使用的就是這個(gè)名稱,注意這里指定的文件名不可以包含路徑,因?yàn)樗械奈募际悄J(rèn)存儲(chǔ)到/data/data/<packagename>/files/目錄下的。第二個(gè)參數(shù)是文件的操作模式,主要有兩種模式可選,MODE_PRIVATE和MODE_APPEND。其中其中MODE_PRIVATE是默認(rèn)的操作模式,表示當(dāng)指定同樣文件名的時(shí)候,所寫入的內(nèi)容將會(huì)覆蓋原文件中的內(nèi)容,而MODE_APPEND則表示如果該文件已存在,就往文件里面追加內(nèi)容,不存在就創(chuàng)建新文件。
openFileOutput()方法返回的是一個(gè)FileOutputStream對(duì)象,得到了這個(gè)對(duì)象之后,就可以使用Java流的方式將數(shù)據(jù)寫入到文件中了。
下面我就簡(jiǎn)單地寫個(gè)demo來(lái)實(shí)現(xiàn)一下最基本的文件存儲(chǔ)。
首先布局文件里放入一個(gè)EditText控件和Buttton控件,EditText控件用于輸入需要保留的數(shù)據(jù),Button控件用于響應(yīng)點(diǎn)擊監(jiān)聽(tīng)。
activity_main.xml代碼:
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"android:layout_width="match_parent"android:layout_height="match_parent"android:orientation="vertical"><EditTextandroid:id="@+id/et"android:layout_width="match_parent"android:layout_height="wrap_content"android:gravity="center"android:hint="請(qǐng)輸入需要保存到文件的數(shù)據(jù)"android:textSize="18sp" /><Buttonandroid:id="@+id/btn"android:layout_width="match_parent"android:layout_height="wrap_content"android:layout_marginTop="10dp"android:text="save"android:textAllCaps="false"android:textSize="20sp"android:textStyle="bold" /> </LinearLayout>MainActivity.java代碼:
package com.example.administrator.filesave;import android.support.v7.app.AppCompatActivity; import android.os.Bundle; import android.view.View; import android.widget.Button; import android.widget.EditText; import android.widget.Toast;import java.io.BufferedWriter; import java.io.File; import java.io.FileNotFoundException; import java.io.FileOutputStream; import java.io.IOException; import java.io.OutputStreamWriter;public class MainActivity extends AppCompatActivity {private EditText et;private Button btn;private FileOutputStream out;private BufferedWriter writer;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);initView();}private void initView() {et = (EditText) findViewById(R.id.et);btn = (Button) findViewById(R.id.btn);}@Overrideprotected void onResume() {super.onResume();btn.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View v) {save();//存儲(chǔ)數(shù)據(jù)到文件}});}private void save() {try {out = openFileOutput("data", MODE_PRIVATE);//初始化FileOutput對(duì)象writer = new BufferedWriter(new OutputStreamWriter(out));//初始化BufferedWriter(帶緩存的字符輸出流)try {writer.write(et.getText().toString());//將EditText控件中的數(shù)據(jù)寫入輸出流。Toast.makeText(this, "file already saved", Toast.LENGTH_SHORT).show();} catch (IOException e) {e.printStackTrace();}} catch (FileNotFoundException e) {e.printStackTrace();} finally {if (writer != null) {try {writer.close();//關(guān)閉輸出流} catch (IOException e) {e.printStackTrace();}}}}}自定義save()方法用來(lái)保存數(shù)據(jù)到文件,我們只需要在Button的點(diǎn)擊監(jiān)聽(tīng)事件中調(diào)用save()方法就可以了。
?
效果圖:
?
最后,我們?cè)趺床拍芸吹綌?shù)據(jù)已經(jīng)存入到本地文件中了呢?光看這Toast提示,顯然是不可取的。
這個(gè)文件的具體路徑為data/data/<com.example.administrator.filesave>/files/data,首先我們得打開(kāi)Device File Explorer
然后找data文件,再找一個(gè)data文件,最后找到帶我們項(xiàng)目包名的文件,下面有個(gè)files文件夾,最后有一個(gè)data文件,打開(kāi)data文件,你可以直接看到我們寫入的deal friend數(shù)據(jù),效果圖如下所示:
如果你是一個(gè)足夠細(xì)心的同學(xué),在你多次輸入數(shù)據(jù)之后,會(huì)發(fā)現(xiàn)數(shù)據(jù)只會(huì)一直被新增的數(shù)據(jù)所覆蓋,原因就是我們?cè)贛ainActivity.java文件中的save()方法中,為FileOutput對(duì)象初始化的時(shí)候,構(gòu)造函數(shù)傳參的第一個(gè)參數(shù)是“data”文件名,第二個(gè)參數(shù)為MODE_PRIVATE,這個(gè)操作模式是默認(rèn)的操作模式,表示當(dāng)指定同樣文件名的時(shí)候,所寫入的內(nèi)容將會(huì)覆蓋源文件中的內(nèi)容,如果希望數(shù)據(jù)追加,那么只需要將操作模式設(shè)置為MODE_APPEND即可。
在MainActivity活動(dòng)被銷毀的時(shí)候?qū)⑺矔r(shí)數(shù)據(jù)保存到文件:
這個(gè)功能就很簡(jiǎn)單了,只需要重寫onDestroy()方法,然后在這里面調(diào)用save()方法就可以了。
@Overrideprotected void onDestroy() {super.onDestroy();save();//保存瞬時(shí)數(shù)據(jù)到文件}?
?
從文件中讀取數(shù)據(jù):
上面我們實(shí)現(xiàn)了將數(shù)據(jù)保存到本地文件的功能,現(xiàn)在我們?cè)賹?shí)現(xiàn)一個(gè)從文件中讀取數(shù)據(jù)的功能。
上面我們?cè)贛ainActivity活動(dòng)被銷毀之后,將數(shù)據(jù)保存到本地文件中,那么我們能否在下此打開(kāi)程序的時(shí)候,直接再將數(shù)據(jù)從文件中讀取并顯示在EditText控件中呢?顯然是可以的。
類似于將數(shù)據(jù)存儲(chǔ)到文件中,Context類中還提供了一個(gè)openFileInput()方法,用于從文件中讀取數(shù)據(jù)。這個(gè)方法要比openFileOutput()簡(jiǎn)單一些,它只接收一個(gè)參數(shù),即要讀取的文件名,然后系統(tǒng)會(huì)自動(dòng)到data/data/<package name>/files/目錄下去加載這個(gè)文件,并返回一個(gè)FileInputStream對(duì)象,得到了這個(gè)對(duì)象之后再通過(guò)java流的方式就可以將數(shù)據(jù)讀取出來(lái)了。
MainActivity.java代碼:
package com.example.administrator.filesave;import android.support.v7.app.AppCompatActivity; import android.os.Bundle; import android.view.View; import android.widget.Button; import android.widget.EditText; import android.widget.Toast;import java.io.BufferedReader; import java.io.BufferedWriter; import java.io.File; import java.io.FileInputStream; import java.io.FileNotFoundException; import java.io.FileOutputStream; import java.io.IOException; import java.io.InputStreamReader; import java.io.OutputStreamWriter;public class MainActivity extends AppCompatActivity {private EditText et;private Button btn;private FileOutputStream out;private BufferedWriter writer;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);initView();//初始化控件initData();//初始化數(shù)據(jù)}private void initView() {et = (EditText) findViewById(R.id.et);btn = (Button) findViewById(R.id.btn);}private void initData() {et.setText("" + load());}@Overrideprotected void onResume() {super.onResume();btn.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View v) {save();//存儲(chǔ)數(shù)據(jù)到文件}});}private void save() {try {out = openFileOutput("data", MODE_PRIVATE);//初始化FileOutput對(duì)象writer = new BufferedWriter(new OutputStreamWriter(out));//初始化BufferedWriter(帶緩存的字符輸出流)try {writer.write(et.getText().toString());//將EditText控件中的數(shù)據(jù)寫入輸出流。Toast.makeText(this, "file already saved", Toast.LENGTH_SHORT).show();} catch (IOException e) {e.printStackTrace();}} catch (FileNotFoundException e) {e.printStackTrace();} finally {if (writer != null) {try {writer.close();//關(guān)閉輸出流} catch (IOException e) {e.printStackTrace();}}}}private String load() {FileInputStream in = null;BufferedReader reader = null;StringBuilder content = new StringBuilder();try {in = openFileInput("data");reader = new BufferedReader(new InputStreamReader(in));String line = "";while ((line = reader.readLine()) != null) {content.append(line);}} catch (FileNotFoundException e) {e.printStackTrace();} catch (IOException e) {e.printStackTrace();} finally {if (reader != null) {try {reader.close();} catch (IOException e) {e.printStackTrace();}}}return content.toString();}@Overrideprotected void onDestroy() {super.onDestroy();save();//保存瞬時(shí)數(shù)據(jù)到文件} }在這段代碼中,首先通過(guò)openFileInput()方法獲取到了一個(gè)FileInputStream對(duì)象,然后借助它又構(gòu)建出一個(gè)BufferedReader(帶緩存的字符輸入流)對(duì)象,這樣我們就可以通過(guò)BufferedReader進(jìn)行一行行地讀取,把文件中所有的文本內(nèi)容全部讀取拼接出來(lái),并存放在一個(gè)StringBuilder對(duì)象中,最后將讀取到的內(nèi)容返回就可以了。
效果圖:
輸入“聽(tīng)媽媽的話”,這個(gè)應(yīng)該屬于瞬時(shí)數(shù)據(jù),然后我們退出程序即MainActivity活動(dòng)被銷毀,此時(shí)根據(jù)Activity的生命周期,系統(tǒng)會(huì)調(diào)用onDestroy()方法,然后再去調(diào)用save()方法保留數(shù)據(jù)。
再次打開(kāi)程序,我們發(fā)現(xiàn)數(shù)據(jù)已經(jīng)從文件中讀取出來(lái),并且顯示到EditText控件上了。
?
?
總結(jié)
以上是生活随笔為你收集整理的Android数据存储之文件存储(瞬时数据的存储与读取)项目已上传GitHub的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。
- 上一篇: 解禁对股价的影响 其实不一定就是利空
- 下一篇: Android数据存储之SharedPr