Android系统中的数据存储方式(上)
1.文件存储:以I/O流形式把数据存入手机内存或者SD卡,可以存储大数据,如音乐,图片或者视频等。
2.SharedPreferences:它本质上是一个XML文件,以Map《Object,Object》形式存入手机内存中。常用于存储简单的参数设置,如QQ登陆账号密码的存储,窗口功能状态的存储等,使用起来简单、方便。
3.SQLite数据库:SQLite是一个轻量级,跨平台的数据库。数据库所有信息都存储在单一文件内,占用内存小,并且支持基本SQL语法,是项目中经常被采用的一种数据存储方式,通常用于存储用户信息等。
4.Content Provider:内存提供者,Android四大组件之一,以数据库形式存入手机内存,可以共享自己的数据给其他应用使用。
5.网络存储:把数据存储到服务器,不存储在本地,使用的时候直接从网络获取,避免了手机端信息丢失以及其他的安全隐患。
本文通过一个案例:QQ登陆,来演示文件存储(内存+SD卡)、SharedPreferences的具体用法。
需求分析:
输入qq号码和密码,点击登陆按钮如果用户勾选记住密码,则将账号和密码存储到手机内存。下次进入应用回显账号和密码。
界面布局实现:
<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:gravity="center_horizontal"
android:orientation="vertical"
tools:context=".MainActivity" >
<ImageView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:src="@drawable/qq" />
<EditText
android:id="@+id/et_qq"
android:inputType="number"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:hint="请输入qq号码" />
<EditText
android:id="@+id/et_password"
android:inputType="textPassword"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:hint="请输入密码" />
<CheckBox
android:id="@+id/cb_remember"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:text="记住密码" />
<Button
android:layout_width="200dip"
android:layout_height="wrap_content"
android:onClick="login"
android:text="登陆" />
</LinearLayout>
1.内存存储
当应用安装到Android 后,系统会根据每个应用的包名创建一个/data/data/包名/的文件夹,访问自己包名下的目录是不需要权限的,并且Android 已经提供了非常简便的API 可以直接去访问该文件夹。
QQ登陆案例中,当点击登陆按钮后,如果记住密码则将账号和密码存储到内存中:
import java.io.BufferedReader;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.InputStreamReader;
import android.app.Activity;
import android.os.Bundle;
import android.text.TextUtils;
import android.util.Log;
import android.view.View;
import android.widget.CheckBox;
import android.widget.EditText;
import android.widget.Toast;
public class MainActivity extends Activity {
private static final String TAG = "MainActivity";
private EditText et_qq;
private EditText et_password;
private CheckBox cb_remember;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
et_qq = (EditText) findViewById(R.id.et_qq);
et_password = (EditText) findViewById(R.id.et_password);
cb_remember = (CheckBox) findViewById(R.id.cb_remember);
// 获取原来在文件中保存的qq号码和密码,回显到界面上.
//File file = new File("/data/data/com.itheima.qqlogin/info.txt");
//getFileDir() === /data/data/<当前应用程序包名>/files
File file = new File(this.getCacheDir(), "info.txt");
if (file.exists() && file.length() > 0) {
try {
FileInputStream fis = new FileInputStream(file);
BufferedReader br = new BufferedReader(new InputStreamReader(
fis));
// 10000##abc 10086####ab
String info = br.readLine();
String qq = info.split("##")[0];
String pwd = info.split("##")[1];
et_qq.setText(qq);
et_password.setText(pwd);
} catch (Exception e) {
e.printStackTrace();
}
}
}
public void login(View view) {
String qq = et_qq.getText().toString().trim();
String pwd = et_password.getText().toString().trim();
if (TextUtils.isEmpty(qq) || TextUtils.isEmpty(pwd)) {
Toast.makeText(this, "用户名或者密码不能为空", 0).show();
return;
}
if (cb_remember.isChecked()) {// 记住密码
Log.i(TAG, "记住密码");
try {
File file = new File(getCacheDir(), "info.txt");
FileOutputStream fos = new FileOutputStream(file);
// 10000##abc
fos.write((qq + "##" + pwd).getBytes());
fos.close();
Toast.makeText(this, "数据保存成功", 0).show();
} catch (Exception e) {
e.printStackTrace();
Toast.makeText(this, "数据保存失败", 0).show();
}
} else {// 不需要记住密码
Log.i(TAG, "不需要记住密码");
}
}
}
>注意尽量用getFileDir()来获取/data/data/包名/files/,而不是直接写全路径。
补充:getCacheDir()代表的是/data/data/包名/cache/,缓存目录,保存一些临时的数据,可以随时删掉不影响程序的逻辑。
可以通过File Explorer窗口查看文件是否写入成功:
2.外部存储SD卡
应用程序可以把数据存储在外部存储设备上,也就是常见的SD卡上(该文件通常位于mnt/sdcard目录下,不同产商生产的手机这个路径可能不同),但是在操作sd卡的时候最好去判断下sd卡是否可用以及剩余空间是否足够,因为有的手机可能没有插sd卡。
在QQ登陆案例中当我们登陆的时候如果是存储在sd卡,可以这样操作:
import java.io.BufferedReader;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.InputStreamReader;
import android.app.Activity;
import android.os.Bundle;
import android.os.Environment;
import android.text.TextUtils;
import android.text.format.Formatter;
import android.util.Log;
import android.view.View;
import android.widget.CheckBox;
import android.widget.EditText;
import android.widget.Toast;
public class MainActivity extends Activity {
private static final String TAG = "MainActivity";
private EditText et_qq;
private EditText et_password;
private CheckBox cb_remember;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
et_qq = (EditText) findViewById(R.id.et_qq);
et_password = (EditText) findViewById(R.id.et_password);
cb_remember = (CheckBox) findViewById(R.id.cb_remember);
// 获取原来在文件中保存的qq号码和密码,回显到界面上.
File file = new File(Environment.
getExternalStorageDirectory(),"info.txt");
if (file.exists() && file.length() > 0) {
try {
FileInputStream fis = new FileInputStream(file);
BufferedReader br =
new BufferedReader(new InputStreamReader(fis));
// 10000##abc
String info = br.readLine();
String qq = info.split("##")[0];
String pwd = info.split("##")[1];
et_qq.setText(qq);
et_password.setText(pwd);
} catch (Exception e) {
e.printStackTrace();
}
}
}
public void login(View view) {
String qq = et_qq.getText().toString().trim();
String pwd = et_password.getText().toString().trim();
if (TextUtils.isEmpty(qq) || TextUtils.isEmpty(pwd)) {
Toast.makeText(this,"用户名或者密码不能为空",0).show();
return;
}
if (cb_remember.isChecked()) {// 记住密码
Log.i(TAG, "记住密码");
try {
// 检查sd是否存在,是否可用.
if (!Environment.getExternalStorageState().equals(
Environment.MEDIA_MOUNTED)) {
Toast.makeText(this,
"sd卡不可用,请检查sd卡的状态", 0).show();
return;
}
// 检查sd卡的可用空间.
long size = Environment.
getExternalStorageDirectory() .getFreeSpace();
String info = Formatter.formatFileSize(this, size);
Toast.makeText(this, "可用空间:" + info, 0).show();
File file = new File(Environment.
getExternalStorageDirectory(), "info.txt");
FileOutputStream fos = new FileOutputStream(file);
// 10000##abc
fos.write((qq + "##" + pwd).getBytes());
fos.close();
Toast.makeText(this, "数据保存成功", 0).show();
} catch (Exception e) {
e.printStackTrace();
Toast.makeText(this, "数据保存失败", 0).show();
}
} else {// 不需要记住密码
Log.i(TAG, "不需要记住密码");
}
}
}
最后记得在清单文件中添加读写sd卡的权限
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"/>
3. SharedPrefrerences存储
像QQ登陆这种案例,需要保存用户名,密码,以及一些自定义参数的设置等,使用SharedPrefrerences进行存储是最方便也是最常用的,存储的时候通过key/value键值对的形式保存在XML文件中,该文件位于:data/data/包名/shared_prefs文件夹中。使用的时候,首先需要通过context.getSharedPrefrences(String name,int mode)获取SharedPrefrences的实例对象,在Activity中可以直接使用this代表上下文,如果不是在activity中需要使用Context上下文来获取。
SharedPreferences sp = this.getSharedPreferences(“config”, 0);
第一个参数config代表的是该sp文件的名字第二个参数0代表的是sp文件的操作模式:私有模式MODE_PRIVATE,代表只有本应用程序可以读写。也有其他的一些模式比如:MODE_WORLD_READABLEMODE_WORLD_WRITEABLE指定该SharedPreferences中的数据可以被其他应用程序读写。SharedPreferences对象本身只能获取数据,并不支持数据的存储和修改,数据的存储修改需要通过SharedPreferences.Editor()对象实现
import android.app.Activity;
import android.content.SharedPreferences;
import android.content.SharedPreferences.Editor;
import android.os.Bundle;
import android.text.TextUtils;
import android.util.Log;
import android.view.View;
import android.widget.CheckBox;
import android.widget.EditText;
import android.widget.Toast;
public class MainActivity extends Activity {
private static final String TAG = "MainActivity";
private EditText et_qq;
private EditText et_password;
private CheckBox cb_remember;
// 1.声明sp
private SharedPreferences sp;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
et_qq = (EditText) findViewById(R.id.et_qq);
et_password = (EditText) findViewById(R.id.et_password);
cb_remember = (CheckBox) findViewById(R.id.cb_remember);
// 2.获取到一个参数
sp = this.getSharedPreferences("config", 0);
String qq = sp.getString("qq", "");
String pwd = sp.getString("pwd", "");
et_qq.setText(qq);
et_password.setText(pwd);
}
public void login(View view) {
String qq = et_qq.getText().toString().trim();
String pwd = et_password.getText().toString().trim();
if (TextUtils.isEmpty(qq) || TextUtils.isEmpty(pwd)) {
Toast.makeText(this,
"用户名或者密码不能为空", 0).show();
return;
}
if (cb_remember.isChecked()) {// 记住密码
Log.i(TAG, "记住密码");
// 3.得到sp文件的编辑器
Editor editor = sp.edit();
editor.putString("qq", qq);
editor.putString("pwd", pwd);
// 4.保存数据完毕,一定要记得调用commit方法
editor.commit();
} else {// 不需要记住密码
Log.i(TAG, "不需要记住密码");
}
}
}