数据存储与访问
很多时候我们开发的软件需要对处理后的数据进行访问,以供再次访问。
Android为数据存储提供了如下几种方式:
文件
SharedPreferences(参数)
SQLite函数库
内容提供者(Content provider)
网络
界面:
main.xml
1
<?
xml version="1.0" encoding="utf-8"
?>
2 < LinearLayout xmlns:android ="http://schemas.android.com/apk/res/android"
3 android:layout_width ="fill_parent"
4 android:layout_height ="fill_parent"
5 android:orientation ="vertical" >
6
7 < TextView
8 android:layout_width ="fill_parent"
9 android:layout_height ="wrap_content"
10 android:text ="@string/filename"
11 />
12 < EditText
13 android:layout_width ="fill_parent"
14 android:layout_height ="wrap_content"
15 android:inputType ="text"
16 android:id ="@+id/filename"
17 />
18 < TextView
19 android:layout_width ="fill_parent"
20 android:layout_height ="wrap_content"
21 android:text ="@string/filecontent"
22 />
23 < EditText
24 android:layout_width ="fill_parent"
25 android:layout_height ="wrap_content"
26 android:minLines ="3"
27 android:inputType ="text"
28 android:id ="@+id/filecontent"
29 />
30 < Button
31 android:layout_width ="wrap_content"
32 android:layout_height ="wrap_content"
33 android:text ="@string/button"
34 android:id ="@+id/button"
35 />
36
37 </ LinearLayout >
2 < LinearLayout xmlns:android ="http://schemas.android.com/apk/res/android"
3 android:layout_width ="fill_parent"
4 android:layout_height ="fill_parent"
5 android:orientation ="vertical" >
6
7 < TextView
8 android:layout_width ="fill_parent"
9 android:layout_height ="wrap_content"
10 android:text ="@string/filename"
11 />
12 < EditText
13 android:layout_width ="fill_parent"
14 android:layout_height ="wrap_content"
15 android:inputType ="text"
16 android:id ="@+id/filename"
17 />
18 < TextView
19 android:layout_width ="fill_parent"
20 android:layout_height ="wrap_content"
21 android:text ="@string/filecontent"
22 />
23 < EditText
24 android:layout_width ="fill_parent"
25 android:layout_height ="wrap_content"
26 android:minLines ="3"
27 android:inputType ="text"
28 android:id ="@+id/filecontent"
29 />
30 < Button
31 android:layout_width ="wrap_content"
32 android:layout_height ="wrap_content"
33 android:text ="@string/button"
34 android:id ="@+id/button"
35 />
36
37 </ LinearLayout >
string.xml
1
<?
xml version="1.0" encoding="utf-8"
?>
2 < resources >
3
4 < string name ="filename" > 文件名称 </ string >
5 < string name ="app_name" > 文件操作 </ string >
6 < string name ="filecontent" > 文件内容 </ string >
7 < string name ="button" > 保存 </ string >
8 < string name ="success" > 保存成功! </ string >
9 < string name ="fail" > 保存失败! </ string >
10 </ resources >
2 < resources >
3
4 < string name ="filename" > 文件名称 </ string >
5 < string name ="app_name" > 文件操作 </ string >
6 < string name ="filecontent" > 文件内容 </ string >
7 < string name ="button" > 保存 </ string >
8 < string name ="success" > 保存成功! </ string >
9 < string name ="fail" > 保存失败! </ string >
10 </ resources >
FileActivity.java
1
package
com.gaolei.file;
2
3 import android.app.Activity;
4 import android.os.Bundle;
5 import android.view.View;
6 import android.view.View.OnClickListener;
7 import android.widget.Button;
8 import android.widget.EditText;
9 import android.widget.Toast;
10
11 import com.gaolei.service.FileService;
12
13 public class FileActivity extends Activity {
14 private Button button;
15
16 /** Called when the activity is first created. */
17 @Override
18 public void onCreate(Bundle savedInstanceState) {
19 super .onCreate(savedInstanceState);
20 setContentView(R.layout.main);
21 button = (Button) this .findViewById(R.id.button);
22 button.setOnClickListener( new ButtonclckListener());
23 }
24
25 public final class ButtonclckListener implements OnClickListener {
26
27 @Override
28 public void onClick(View v) {
29 EditText filenameText = (EditText) findViewById(R.id.filename);
30 EditText filecontentText = (EditText) findViewById(R.id.filecontent);
31 String filename = filenameText.getText().toString();
32 String content = filecontentText.getText().toString();
33
34 FileService fileService = new FileService(getApplicationContext());
35 try {
36 fileService.save(filename, content);
37 Toast.makeText(getApplicationContext(), R.string.success, 1 )
38 .show();
39 } catch (Exception e) {
40 Toast.makeText(getApplicationContext(), R.string.fail, 1 )
41 .show();
42 e.printStackTrace();
43 }
44
45 }
46
47 }
48 }
2
3 import android.app.Activity;
4 import android.os.Bundle;
5 import android.view.View;
6 import android.view.View.OnClickListener;
7 import android.widget.Button;
8 import android.widget.EditText;
9 import android.widget.Toast;
10
11 import com.gaolei.service.FileService;
12
13 public class FileActivity extends Activity {
14 private Button button;
15
16 /** Called when the activity is first created. */
17 @Override
18 public void onCreate(Bundle savedInstanceState) {
19 super .onCreate(savedInstanceState);
20 setContentView(R.layout.main);
21 button = (Button) this .findViewById(R.id.button);
22 button.setOnClickListener( new ButtonclckListener());
23 }
24
25 public final class ButtonclckListener implements OnClickListener {
26
27 @Override
28 public void onClick(View v) {
29 EditText filenameText = (EditText) findViewById(R.id.filename);
30 EditText filecontentText = (EditText) findViewById(R.id.filecontent);
31 String filename = filenameText.getText().toString();
32 String content = filecontentText.getText().toString();
33
34 FileService fileService = new FileService(getApplicationContext());
35 try {
36 fileService.save(filename, content);
37 Toast.makeText(getApplicationContext(), R.string.success, 1 )
38 .show();
39 } catch (Exception e) {
40 Toast.makeText(getApplicationContext(), R.string.fail, 1 )
41 .show();
42 e.printStackTrace();
43 }
44
45 }
46
47 }
48 }
使用文件进行数据存储
首先给大家介绍使用文件如何对数据进行存储,Activity提供了openFileOutput()方法可以用于把数据输出到文件中,具体的实现过程与在J2SE环境中保存数据到文件中是一样的。
public class FileActivity extends Activity {
@Override public void onCreate(Bundle savedInstanceState) {
...
FileOutputStream outStream = this.openFileOutput("itcast.txt", Context.MODE_PRIVATE);
outStream.write("gaolei".getBytes());
outStream.close();
}
}
openFileOutput()方法的第一参数用于指定文件名称,不能包含路径分隔符“/” ,如果文件不存在,Android 会自动创建它。创建的文件保存在/data/data/<package name>/files目录,如: /data/data/cn.itcast.action/files/itcast.txt ,通过点击Eclipse菜单“Window”-“Show View”-“Other”,在对话窗口中展开android文件夹,选择下面的File Explorer视图,然后在File Explorer视图中展开/data/data/<package name>/files目录就可以看到该文件。
openFileOutput()方法的第二参数用于指定操作模式,有四种模式,分别为: Context.MODE_PRIVATE = 0
Context.MODE_APPEND = 32768
Context.MODE_PRIVATE:为默认操作模式,代表该文件是私有数据,只能被应用本身访问,在该模式下,写入的内容会覆盖原文件的内容,如果想把新写入的内容追加到原文件中。可以使用Context.MODE_APPEND
Context.MODE_APPEND:模式会检查文件是否存在,存在就往文件追加内容,否则就创建新文件。
Context.MODE_WORLD_READABLE和Context.MODE_WORLD_WRITEABLE用来控制其他应用是否有权限读写该文件。
MODE_WORLD_READABLE:表示当前文件可以被其他应用读取;MODE_WORLD_WRITEABLE:表示当前文件可以被其他应用写入。
如果希望文件被其他应用读和写,可以传入:
openFileOutput("itcast.txt", Context.MODE_WORLD_READABLE + Context.MODE_WORLD_WRITEABLE);
android有一套自己的安全模型,当应用程序(.apk)在安装时系统就会分配给他一个userid,当该应用要去访问其他资源比如文件的时候,就需要userid匹配。默认情况下,任何应用创建的文件,sharedpreferences,数据库都应该是私有的(位于/data/data/<package name>/files),其他程序无法访问。除非在创建时指定了Context.MODE_WORLD_READABLE或者Context.MODE_WORLD_WRITEABLE ,只有这样其他程序才能正确访问。
业务类:
1
package
com.gaolei.service;
2
3 import java.io.ByteArrayOutputStream;
4 import java.io.FileInputStream;
5 import java.io.FileOutputStream;
6 import java.io.IOException;
7
8 import android.content.Context;
9
10 public class FileService {
11
12 private Context context;
13
14 public FileService(Context context) {
15 this .context = context;
16 }
17
18 /**
19 * 保存文件
20 *
21 * @param filename
22 * 文件名称
23 * @param content
24 * 文件内容
25 * @throws IOException
26 */
27 public void save(String filename, String content) throws Exception {
28 // 私有操作模式:创建出来的文件只能被本应用访问,其他应用无法访问该文件,
29 // 另外采用私有操作模式创建的文件,写入文件中的内容会覆盖原文件的内容
30 FileOutputStream outStream = context.openFileOutput(filename,
31 Context.MODE_PRIVATE);
32 outStream.write(filename.getBytes());
33 outStream.close();
34 }
35
36 /**
37 * 读取文件内容
38 *
39 * @param filename
40 * 文件名称
41 * @return 文件内容
42 * @throws Exception
43 */
44 public String read(String filename) throws Exception {
45 FileInputStream inStream = context.openFileInput(filename);
46 byte [] buffer = new byte [ 1024 ];
47 ByteArrayOutputStream outStream = new ByteArrayOutputStream();
48 int len = 0 ;
49 while ((len = inStream.read(buffer)) != - 1 ) {
50 outStream.write(buffer, 0 , len);
51 }
52 byte [] data = outStream.toByteArray();
53 outStream.close();
54 inStream.close();
55 return new String(data);
56 }
57
58 }
59
2
3 import java.io.ByteArrayOutputStream;
4 import java.io.FileInputStream;
5 import java.io.FileOutputStream;
6 import java.io.IOException;
7
8 import android.content.Context;
9
10 public class FileService {
11
12 private Context context;
13
14 public FileService(Context context) {
15 this .context = context;
16 }
17
18 /**
19 * 保存文件
20 *
21 * @param filename
22 * 文件名称
23 * @param content
24 * 文件内容
25 * @throws IOException
26 */
27 public void save(String filename, String content) throws Exception {
28 // 私有操作模式:创建出来的文件只能被本应用访问,其他应用无法访问该文件,
29 // 另外采用私有操作模式创建的文件,写入文件中的内容会覆盖原文件的内容
30 FileOutputStream outStream = context.openFileOutput(filename,
31 Context.MODE_PRIVATE);
32 outStream.write(filename.getBytes());
33 outStream.close();
34 }
35
36 /**
37 * 读取文件内容
38 *
39 * @param filename
40 * 文件名称
41 * @return 文件内容
42 * @throws Exception
43 */
44 public String read(String filename) throws Exception {
45 FileInputStream inStream = context.openFileInput(filename);
46 byte [] buffer = new byte [ 1024 ];
47 ByteArrayOutputStream outStream = new ByteArrayOutputStream();
48 int len = 0 ;
49 while ((len = inStream.read(buffer)) != - 1 ) {
50 outStream.write(buffer, 0 , len);
51 }
52 byte [] data = outStream.toByteArray();
53 outStream.close();
54 inStream.close();
55 return new String(data);
56 }
57
58 }
59
读取文件内容
如果要打开存放在/data/data/<package name>/files目录应用私有的文件,可以使用Activity提供openFileInput()方法。
FileInputStream inStream = this.getContext().openFileInput("itcast.txt");
Log.i("FileTest", readInStream(inStream));
或者直接使用文件的绝对路径:
File file = new File("/data/data/cn.itcast.action/files/itcast.txt");
FileInputStream inStream = new FileInputStream(file);
Log.i("FileTest", readInStream(inStream));
注意:上面文件路径中的“cn.itcast.action”为应用所在包,当你在编写代码时应替换为你自己应用使用的包。
对于私有文件只能被创建该文件的应用访问,如果希望文件能被其他应用读和写,可以在创建文件时,指定Context.MODE_WORLD_READABLE和Context.MODE_WORLD_WRITEABLE权限。
Activity还提供了getCacheDir()和getFilesDir()方法:
getCacheDir()方法用于获取/data/data/<package name>/cache目录
getFilesDir()方法用于获取/data/data/<package name>/files目录
Context.MODE_WORLD_READABLE = 1
Context.MODE_WORLD_WRITEABLE = 2
AndroidMainFest.xml
1
<?
xml version="1.0" encoding="utf-8"
?>
2 < manifest xmlns:android ="http://schemas.android.com/apk/res/android"
3 package ="com.gaolei.file"
4 android:versionCode ="1"
5 android:versionName ="1.0" >
6
7 < uses-sdk android:minSdkVersion ="15" />
8
9 < application
10 android:icon ="@drawable/ic_launcher"
11 android:label ="@string/app_name" >
12 < activity
13 android:name =".FileActivity"
14 android:label ="@string/app_name" >
15 < intent-filter >
16 < action android:name ="android.intent.action.MAIN" />
17
18 < category android:name ="android.intent.category.LAUNCHER" />
19 </ intent-filter >
20 </ activity >
21 < uses-library android:name ="android.test.runner" />
22 </ application >
23 < instrumentation android:name ="android.test.InstrumentationTestRunner"
24 android:targetPackage ="com.gaolei.file" android:label ="Tests for My Apps" >
25 </ instrumentation >
26
27 </ manifest >
2 < manifest xmlns:android ="http://schemas.android.com/apk/res/android"
3 package ="com.gaolei.file"
4 android:versionCode ="1"
5 android:versionName ="1.0" >
6
7 < uses-sdk android:minSdkVersion ="15" />
8
9 < application
10 android:icon ="@drawable/ic_launcher"
11 android:label ="@string/app_name" >
12 < activity
13 android:name =".FileActivity"
14 android:label ="@string/app_name" >
15 < intent-filter >
16 < action android:name ="android.intent.action.MAIN" />
17
18 < category android:name ="android.intent.category.LAUNCHER" />
19 </ intent-filter >
20 </ activity >
21 < uses-library android:name ="android.test.runner" />
22 </ application >
23 < instrumentation android:name ="android.test.InstrumentationTestRunner"
24 android:targetPackage ="com.gaolei.file" android:label ="Tests for My Apps" >
25 </ instrumentation >
26
27 </ manifest >
测试类:
1
package
com.gaolei.test;
2
3 import com.gaolei.service.FileService;
4
5 import android.test.AndroidTestCase;
6 import android.util.Log;
7
8 public class FileServiceTest extends AndroidTestCase {
9
10 private static final String TAG = " FileServiceTest " ;
11
12 public void testRead() throws Throwable {
13 FileService fileService = new FileService( this .getContext());
14 String result = fileService.read( " gaolei.txt " );
15 Log.i(TAG, result);
16 }
17 }
2
3 import com.gaolei.service.FileService;
4
5 import android.test.AndroidTestCase;
6 import android.util.Log;
7
8 public class FileServiceTest extends AndroidTestCase {
9
10 private static final String TAG = " FileServiceTest " ;
11
12 public void testRead() throws Throwable {
13 FileService fileService = new FileService( this .getContext());
14 String result = fileService.read( " gaolei.txt " );
15 Log.i(TAG, result);
16 }
17 }