(内容整理自张泽华教程)
1. 概述
使用文件进行数据存储
首先给大家介绍使用文件如何对数据进行存储,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);
读取文件内容
如果要打开存放在/data/data/<package name>/files目录应用私有的文件,可以使用Activity提供openFileInput()方法。
FileInputStream inStream = this.getContext().openFileInput("itcast.txt");
Log.i("FileTest", readInStream(inStream));
readInStream()的方法请看本页下面备注。
或者直接使用文件的绝对路径:
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目录
把文件存放在SDCard
在程序中访问SDCard,你需要申请访问SDCard的权限。
在AndroidManifest.xml中加入访问SDCard的权限如下:
<!-- 在SDCard中创建与删除文件权限-->
<uses-permission android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS"/>
<!-- 往SDCard写入数据权限-->
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
要往SDCard存放文件,程序必须先判断手机是否装有SDCard,并且可以进行读写。
注意:访问SDCard必须在AndroidManifest.xml中加入访问SDCard的权限
if(Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED)){
File sdCardDir = Environment.getExternalStorageDirectory();//获取SDCard目录
File saveFile = new File(sdCardDir, “itcast.txt”);
FileOutputStream outStream = new FileOutputStream(saveFile);
outStream.write("传智播客".getBytes());
outStream.close();
}
Environment.getExternalStorageState()方法用于获取SDCard的状态,如果手机装有SDCard,并且可以进行读写,那么方法返回的状态等于Environment.MEDIA_MOUNTED。
Environment.getExternalStorageDirectory()方法用于获取SDCard的目录,当然要获取SDCard的目录,你也可以这样写:
File sdCardDir = new File("/mnt/sdcard"); //获取SDCard目录
File saveFile = new File(sdCardDir, "itcast.txt");
//上面两句代码可以合成一句:File saveFile = new File("/mnt/sdcard/itcast.txt");
FileOutputStream outStream = new FileOutputStream(saveFile);
outStream.write("传智播客test".getBytes());
outStream.close();
2. 示例代码
获取SD卡大小示例代码
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
TextView tv = (TextView) findViewById(R.id.tv_sdsize);
File path = Environment.getExternalStorageDirectory();
StatFs stat = new StatFs(path.getPath());
long blockSize = stat.getBlockSize();
long availableBlocks = stat.getAvailableBlocks();
long sizeAvailSize = blockSize * availableBlocks;
String str = Formatter.formatFileSize(this, sizeAvailSize);
tv.setText(str);
}
数据存储示例代码:
activity_main.xml 布局
<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"
tools:context=".MainActivity" >
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="@string/input_name" />
<EditText
android:id="@+id/et_name"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:inputType="text" />
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="@string/input_password" />
<EditText
android:id="@+id/et_password"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:inputType="textPassword" />
<RelativeLayout
android:layout_width="fill_parent"
android:layout_height="wrap_content" >
<CheckBox
android:id="@+id/cb_remember"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="@string/remember_pwd" />
<Button
android:onClick="login"
android:layout_alignParentRight="true"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="@string/login" />
</RelativeLayout>
<RadioGroup
android:id="@+id/rg_save_location"
android:orientation="horizontal"
android:layout_width="wrap_content"
android:layout_height="wrap_content" >
<RadioButton
android:id="@+id/rb_rom"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:checked="true"
android:text="ROM" />
<RadioButton
android:id="@+id/rb_sd"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="SD" />
<RadioButton
android:id="@+id/rb_sp"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="SP" />
</RadioGroup>
</LinearLayout>
AndroidManifest.xml SD权限
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.itheima.login"
android:versionCode="1"
android:versionName="1.0" >
<uses-sdk
android:minSdkVersion="8"
android:targetSdkVersion="17" />
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
<application
android:allowBackup="true"
android:icon="@drawable/ic_launcher"
android:label="@string/app_name"
android:theme="@style/AppTheme" >
<activity
android:name="com.itheima.login.MainActivity"
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>
SaveService.java 业务类
package com.itheima.login.service;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.util.HashMap;
import java.util.Map;
import android.content.Context;
import android.content.SharedPreferences;
import android.content.SharedPreferences.Editor;
import android.os.Environment;
import com.itheima.login.utils.StreamTools;
public class SaveService {
/**
* 保存数据到手机rom的文件里面.
* @param context 应用程序的上下文 提供环境
* @param name 用户名
* @param password 密码
* @throws Exception
*/
public static void saveToRom(Context context, String name , String password) throws Exception{
//File file = new File("/data/data/com.itheima.login/files/pwd.txt");
File file = new File(context.getFilesDir(),"pwd.txt");
FileOutputStream fos = new FileOutputStream(file);
String txt = name+":"+password;
fos.write(txt.getBytes());
fos.flush();
fos.close();
}
/**
* 从rom读取用户的密码信息
* @param context
* @return
*/
public static String readFromRom(Context context) throws Exception{
File file = new File(context.getFilesDir(),"pwd.txt");
FileInputStream fis = new FileInputStream(file);
String result = StreamTools.readFromStream(fis);
return result;
}
/**
* 保存数据到SD卡
* @param name 用户名
* @param password 密码
* @throws Exception
*/
public static void saveTOSD(String name ,String password) throws Exception{
File file = new File(Environment.getExternalStorageDirectory(),"pwd.txt");
FileOutputStream fos = new FileOutputStream(file);
String txt = name+":"+password;
fos.write(txt.getBytes());
fos.flush();
fos.close();
}
/**
* 保存应用程序数据 到sharedpreference
* @param context 上下文
* @param name 姓名
* @param password 密码
*/
public static void saveTOSP (Context context, String name, String password){
//获取系统的一个sharedpreference文件 名字叫 sp
SharedPreferences sp = context.getSharedPreferences("sp", Context.MODE_WORLD_READABLE+Context.MODE_WORLD_WRITEABLE);
//创建一个编辑器 可以编辑 sp
Editor editor = sp.edit();
editor.putString("name", name);
editor.putString("password", password);
editor.putBoolean("boolean", false);
editor.putInt("int", 8888);
editor.putFloat("float",3.14159f);
//注意:调用 commit 提交 数据到文件.
editor.commit();
//editor.clear();
}
/**
* 获取系统sharepreference里面的数据
* @param context
* @return
*/
public static Map<String,String> readFromSP(Context context){
Map<String,String> map = new HashMap<String, String>();
SharedPreferences sp = context.getSharedPreferences("sp", Context.MODE_PRIVATE);
String name = sp.getString("name", "defaultname");
String password = sp.getString("password", "password");
map.put("name", name);
map.put("password", password);
return map;
}
}
MainActivity.java
package com.itheima.login;
import java.util.Map;
import com.itheima.login.service.SaveService;
import android.os.Bundle;
import android.os.Environment;
import android.app.Activity;
import android.text.TextUtils;
import android.util.Log;
import android.view.View;
import android.widget.CheckBox;
import android.widget.EditText;
import android.widget.RadioGroup;
import android.widget.Toast;
public class MainActivity extends Activity {
private static final String TAG = "MainActivity";
private EditText et_name;
private EditText et_password;
private CheckBox cb_remember;
private RadioGroup rg_save_location;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
et_name = (EditText) findViewById(R.id.et_name);
et_password = (EditText) findViewById(R.id.et_password);
cb_remember = (CheckBox) findViewById(R.id.cb_remember);
rg_save_location = (RadioGroup) findViewById(R.id.rg_save_location);
// 检查 rom文件里面是否 有 密码信息
/*try {
String result = SaveService.readFromRom(this);
String[] infos = result.split(":");
et_name.setText(infos[0]);
et_password.setText(infos[1]);
} catch (Exception e) {
e.printStackTrace();
}*/
//获取sp里面的数据
Map<String, String> map = SaveService.readFromSP(this);
et_name.setText(map.get("name"));
et_password.setText(map.get("password"));
}
public void login(View view) {
String name = et_name.getText().toString().trim();
String password = et_password.getText().toString().trim();
if (TextUtils.isEmpty(name) || TextUtils.isEmpty(password)) {
Toast.makeText(this, R.string.error_msg, Toast.LENGTH_SHORT).show();
return;
}
if (cb_remember.isChecked()) {
Log.i(TAG, "记住密码");
Log.d(TAG, "NAME=" + name);
Log.d(TAG, "PASSWORD=" + password);
int radiobuttonId = rg_save_location.getCheckedRadioButtonId();
switch (radiobuttonId) {
case R.id.rb_rom:
try {
SaveService.saveToRom(this, name, password);
Toast.makeText(this, "保存rom用户名密码 成功", Toast.LENGTH_LONG)
.show();
} catch (Exception e) {
e.printStackTrace();
Toast.makeText(this, "保存rom用户名密码 失败", Toast.LENGTH_LONG)
.show();
}
break;
case R.id.rb_sd:
if (Environment.MEDIA_MOUNTED.equals(Environment
.getExternalStorageState())) {
try {
SaveService.saveTOSD(name, password);
Toast.makeText(this, "保存sd用户名密码 成功", Toast.LENGTH_LONG)
.show();
} catch (Exception e) {
e.printStackTrace();
Toast.makeText(this, "保存sd用户名密码 失败", Toast.LENGTH_LONG)
.show();
}
}else{
Toast.makeText(this, "sd卡不可用 ,请检查sd卡状态", 0).show();
}
break;
case R.id.rb_sp:
SaveService.saveTOSP(this, name, password);
Toast.makeText(this, "保存到sp完成", 0).show();
break;
}
} else {
Log.i(TAG, "不需要记住密码");
}
}
}
------------------- sharePreference 可以使用 PreferenceScreen 通过xml配置文件生成
activity文件
package com.example.datastoragetest;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import org.apache.http.client.methods.HttpGet;
import android.os.Bundle;
import android.os.Environment;
import android.preference.Preference;
import android.preference.Preference.OnPreferenceClickListener;
import android.preference.PreferenceActivity;
import android.preference.PreferenceFragment;
import android.preference.PreferenceManager;
import android.app.Activity;
import android.content.ComponentName;
import android.content.Context;
import android.content.Intent;
import android.util.Log;
import android.view.Menu;
import android.widget.Toast;
public class MainActivity extends Activity {
//方式一: 已过时
// @Override
// protected void onCreate(Bundle savedInstanceState) {
// super.onCreate(savedInstanceState);
// //setContentView(R.layout.activity_main);
//
// addPreferencesFromResource(R.xml.preferencedata1);
// Preference preference = findPreference("orientation");
// preference.setOnPreferenceClickListener(new OnPreferenceClickListener() {
// @Override
// public boolean onPreferenceClick(Preference preference) {
// Toast.makeText(MainActivity.this, "orientation click", Toast.LENGTH_SHORT).show();
// return false;
// }
// });
//
// }
//方式二: 使用fragment
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
//R.id.content为布局文件中的一个ViewGroup
getFragmentManager().beginTransaction().replace(R.id.content, new Prefs2Fragment()).commit();
//文件的读写
Log.i("myLog", getFilesDir().getPath());
Log.i("myLog", Environment.getExternalStorageState());
try {
FileOutputStream fos = openFileOutput("text.txt", Context.MODE_WORLD_WRITEABLE);
fos.write("hello wold\n".getBytes());
fos.close();
} catch (Exception e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
public static class Prefs2Fragment extends PreferenceFragment {
@Override
public void onCreate(Bundle savedInstanceState) {
// TODO Auto-generated method stub
super.onCreate(savedInstanceState);
//设置sharepreference的资源文件
addPreferencesFromResource(R.xml.preferencedata1);
Preference preference = findPreference("orientation");
//监听事件
preference.setOnPreferenceClickListener(new OnPreferenceClickListener() {
@Override
public boolean onPreferenceClick(Preference preference) {
Log.i("myLog", "click event");
return false;
}
});
}
}
}
res/xml/preferencedata1.xml PreferenceScreen 配置文件
其中WIFI可以通过添加intent 直接开启其他的activity
<?xml version="1.0" encoding="utf-8"?>
<PreferenceScreen xmlns:android="http://schemas.android.com/apk/res/android" >
<PreferenceCategory android:title="屏幕">
<CheckBoxPreference
android:key="orientation"
android:title="屏幕方向"
android:summary="设置你屏幕的方向" />
<ListPreference
android:key="resolution "
android:title="分辨率"
android:summary="设置你要的分辨率"
android:dialogTitle="选择分辨率"
android:entries="@array/resolution"
android:entryValues="@array/resolutionValue" />
<RingtonePreference
android:key="myRingtone"
android:title="手机铃声" />
</PreferenceCategory>
<PreferenceScreen android:title="WIFI">
<intent android:action="android.settings.WIFI_SETTINGS"> </intent>
</PreferenceScreen>
<PreferenceScreen android:title="其他">
<PreferenceCategory android:title="wifi">
<EditTextPreference android:title="设置wifi地址" android:key="wifiaddr"/>
<EditTextPreference android:title="设置wifi密码" android:key="wifipasswd" />
</PreferenceCategory>
</PreferenceScreen>
</PreferenceScreen>