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号码和密码,点击登陆按钮如果用户勾选记住密码,则将账号和密码存储到手机内存。下次进入应用回显账号和密码。

android数据存储的五种方式 android 的数据存储方式_android数据存储的五种方式

界面布局实现:

<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窗口查看文件是否写入成功:

android数据存储的五种方式 android 的数据存储方式_android_02

2.外部存储SD卡

应用程序可以把数据存储在外部存储设备上,也就是常见的SD卡上(该文件通常位于mnt/sdcard目录下,不同产商生产的手机这个路径可能不同),但是在操作sd卡的时候最好去判断下sd卡是否可用以及剩余空间是否足够,因为有的手机可能没有插sd卡。

android数据存储的五种方式 android 的数据存储方式_android_03


在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"/>

android数据存储的五种方式 android 的数据存储方式_Text_04

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_READABLE
MODE_WORLD_WRITEABLE
指定该SharedPreferences中的数据可以被其他应用程序读写。

android数据存储的五种方式 android 的数据存储方式_Text_05


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, "不需要记住密码");
            }
        }
    }