很多时候我们的软件需要对处理后的数据进行存储或再次访问。Android为数据存储提供了多种方式,分别有如下几种:
- 文件
- SharedPreferences
- SQLite数据库
- 内容提供者(Content provider)
- 网络
Rom:随机存储器--断电不能保持
rom可以保存
存储
首先给大家介绍使用文件如何对数据进行存储,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("赵雅智博客".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_WORLD_READABLE = 1
- Context.MODE_WORLD_WRITEABLE = 2
- 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 ,只有这样其他程序才能正确访问。
读取
如果要打开存放在/data/data/<package name>/files目录应用私有的文件,可以使用Activity提供openFileInput()方法。
FileInputStream inStream = this.getContext().openFileInput("itcast.txt");
Log.i("FileTest", readInStream(inStream));
readInStream()的方法
public static String readInStream(FileInputStream inStream){
try {
ByteArrayOutputStream outStream = new ByteArrayOutputStream();
byte[] buffer = new byte[1024];
int length = -1;
while((length = inStream.read(buffer)) != -1 ){
outStream.write(buffer, 0, length);
}
outStream.close();
inStream.close();
return outStream.toString();
} catch (IOException e) {
Log.i("FileTest", e.getMessage());
}
return null;
}
或者直接使用文件的绝对路径:
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目录
实例:login
AndroidManifest.xml
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.example.lession02_login"
android:versionCode="1"
android:versionName="1.0" >
<uses-sdk
android:minSdkVersion="8"
android:targetSdkVersion="17" />
<application
android:allowBackup="true"
android:icon="@drawable/ic_launcher"
android:label="@string/app_name"
android:theme="@style/AppTheme" >
<activity
android:name="com.example.lession02_login.LoginActivity"
android:label="@string/app_name" >
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
</application>
</manifest>
布局文件:
<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:orientation="vertical" >
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content" >
<TextView
android:id="@+id/view_name"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="@string/login_name" />
<EditText
android:id="@+id/edit_name"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_weight="1"
android:ems="10"
android:inputType="textPersonName" >
<requestFocus />
</EditText>
</LinearLayout>
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content" >
<TextView
android:id="@+id/view_pass"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="@string/login_pass" />
<EditText
android:id="@+id/edit_pass"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_weight="1"
android:ems="10"
android:inputType="textPersonName" >
<requestFocus />
</EditText>
</LinearLayout>
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content" >
<Button
android:id="@+id/button_login"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="@string/login_login" />
<CheckBox
android:id="@+id/check_remember"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_marginLeft="100dp"
android:text="@string/login_remember" />
</LinearLayout>
</LinearLayout>
strings.xml
<?xml version="1.0" encoding="utf-8"?>
<resources>
<string name="app_name">lession02_login</string>
<string name="action_settings">Settings</string>
<string name="hello_world">Hello world!</string>
<string name="login_name">用户名</string>
<string name="login_pass">密码</string>
<string name="login_login">登陆</string>
<string name="login_remember">记住密码</string>
</resources>
字节输出流对象:StreamTools.java
package com.example.lession02_login.util;
import java.io.ByteArrayOutputStream;
import java.io.FileInputStream;
public class StreamTools {
public static String getValue(FileInputStream fis) throws Exception {
//字节的输出流对象
ByteArrayOutputStream stream = new ByteArrayOutputStream();
byte[] buffer = new byte[1024];
int length = -1;
while ((length = fis.read(buffer)) != -1) {
stream.write(buffer, 0, length);
}
stream.flush();
stream.close();
String value = stream.toString();
return value;
}
}
LoginService.java
package com.example.lession02_login.service;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.util.HashMap;
import java.util.Map;
import com.example.lession02_login.util.StreamTools;
import android.content.Context;
public class LoginService {
// 上下文对象
public Context context;
//通过上下文对象传过来
public LoginService(Context context) {
super();
this.context = context;
}
/**
* 往手机内存上存储用户名与密码的操作
* @param name
* @param pass
* @param fileName
* @return
*/
public boolean saveToRom(String name, String pass, String fileName) {
// 上下文对象的api
try {
// 通过openFileOutput()方法获取一个文件的输出流对象
//MODE_PRIVATE私有模式,只能被应用本身访问,
/* FileOutputStream fos = context.openFileOutput(fileName,
Context.MODE_PRIVATE);*/
//MODE_APPEND追加
FileOutputStream fos = context.openFileOutput(fileName,
Context.MODE_APPEND);
// 拼接用户名密码
String result = name + ":" + pass;
// 写入
fos.write(result.getBytes());
fos.flush();
fos.close();
} catch (Exception e) {
e.printStackTrace();
return false;
}
return true;
}
/**
* 读取操作
* @param name
* @param pass
* @param fileName
* @return
*/
public Map<String, String> readFile(String fileName) {
Map<String, String> map = null;// new HashMap<String, String>();
try {
FileInputStream fis = context.openFileInput(fileName);
String value = StreamTools.getValue(fis);
String values[] = value.split(":");
if (values.length > 0) {
map = new HashMap<String, String>();
map.put("name", values[0]);
map.put("pass", values[1]);
}
} catch (Exception e) {
e.printStackTrace();
}
return map;
}
}
LoginActivity.java
package com.example.lession02_login;
import java.util.Map;
import com.example.lession02_login.service.LoginService;
import android.os.Bundle;
import android.app.Activity;
import android.support.v4.widget.SimpleCursorAdapter.ViewBinder;
import android.text.TextUtils;
import android.view.Menu;
import android.view.View;
import android.widget.Button;
import android.widget.Checkable;
import android.widget.EditText;
import android.widget.Toast;
public class LoginActivity extends Activity {
// 声明获取的用户名和密码
private EditText edit_name, edit_pass;
// 声明登陆按对象
private Button btn_login;
// 声明复选框组件对象
private Checkable box_remember;
// 声明业务对象
private LoginService loginService;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
// 设置显示视图
setContentView(R.layout.activity_login);
// 实例化业务对像
loginService = new LoginService(this);
// 根据id名获取相应组件对象
edit_name = (EditText) findViewById(R.id.edit_name);
edit_pass = (EditText) findViewById(R.id.edit_pass);
btn_login = (Button) findViewById(R.id.button_login);
box_remember = (Checkable) findViewById(R.id.check_remember);
// 给按钮注册事件
btn_login.setOnClickListener(new MyOnclickListener());
//回显数据
/*//私有模式
Map<String,String> map = loginService.readFile("private.txt");*/
Map<String,String> map = loginService.readFile("append.txt");
if(map!=null){
edit_name.setText(map.get("name"));
edit_pass.setText(map.get("pass"));
}
}
@Override
public boolean onCreateOptionsMenu(Menu menu) {
// Inflate the menu; this adds items to the action bar if it is present.
getMenuInflater().inflate(R.menu.login, menu);
return true;
}
// 内部类 有关点击的处理对象
class MyOnclickListener implements View.OnClickListener {
@Override
public void onClick(View v) {
int id = v.getId();
switch (id) {
case R.id.button_login:
// 获取用户名密码
String name = edit_name.getText().toString();
String pass = edit_pass.getText().toString();
// 判断用户名密码是否为空
if (TextUtils.isEmpty(name)) {
Toast.makeText(LoginActivity.this, "用户名不可 为空",
Toast.LENGTH_LONG).show();
return;
} else if (TextUtils.isEmpty(pass)) {
Toast.makeText(LoginActivity.this, " 密码不可 为空",
Toast.LENGTH_LONG).show();
return;
} else {
if (box_remember.isChecked()) {
// 进行保存
// 调用业务对象的业务方法
LoginActivity.this.loginService.saveToRom(name, pass,
"private.txt");
Toast.makeText(LoginActivity.this, "保存用户名和密码",
Toast.LENGTH_LONG).show();
} else {
Toast.makeText(LoginActivity.this, "不保存用户名和密码",
Toast.LENGTH_LONG).show();
}
}
break;
default:
break;
}
}
}
}
D:文件夹
-文件
R:可读
W:可写
X:可执行
三三一组
前三:当前用户,中三:组,后三全局