Android第一行代码(八):数据持久化存储

数据持久化:是指将那些内存中的瞬时数据保存到存储设备中,保证即使在手机或电脑关机的情况下,这些数据仍然不会丢失。持久化技术则提供了一种机制,可以让数据在瞬时状态和持久状态之间进行转换。

Android系统主要提供以下数据存储方式:

  1. 文件存储
  2. SharedPreferences存储
  3. 数据库存储

一:文件存储

默认存储路径:/data/data/<package name>/files/
文件操作模式:MODE_PRIVATE(默认),覆盖; MODE_APPEND:追加

写入文件

可使用openFileOutput()方法,接收两个参数,第一个参数是文件名,第二个参数是文件的操作模式。

public void save(){
  String data = "Date to save";
  FileOutputStream out = null;
  ButteredWriter writer = null;
  try{
        //第一个参数 文件名;第二个参数 文件操作模式; 返回 FileOutputStream对象
        out = openFileOutput("data",Context.MODE_PRIVATE); 
        //实例化一个ButterWriter
        writer = new ButteredWriter(new OutputSreamWriter(out));
        writer.write(data);
  }catch(IOException e){
        e.printStackTrace();
  }finally{
        try{
              if(writer!=null){
                    writer.close();
              }
        }catch(IOException e){
              e.printStackTrace();
   }
}

读取数据

可使用openFileInput()方法,只接受文件名一个参数。

public String load() {
    FileInputStream in = null;

    BufferedReader reader = null;

    StringBuilder content = new StringBuilder();
    try {
        //传入文件名参数,返回FileInputStream对象
        in = openFileInput("data");
        reader = new BufferedReader(new InputStreamReader(in));
        String line = "";
        while ((line = reader.readLine()) != null) {
            content.append(line);
        }
    } catch (IOException e) {
        e.printStackTrace();
    } finally {
        if (reader != null) {
            try {
                reader.close();
            } catch (IOException e) {
                e.printStackTrace();
            }
        }
    }
    return content.toString();
}

二: SharedPreferences存储

默认存储路径:/data/data/<package name>/shared_prefs
操作模式:
MODE_PRIVATE(默认):只有当前的应用程序才能对文件进行读写
MODE_MULTI_PROCESS:用于多个进程对同一个SharedPreferences进行读写。(6.0开始废弃)
存储数据格式:键值对

获取SharedPreferences对象方法:

  1. Context类中的getSharedPreferences()方法,参数一是文件名,参数二是操作模式
  2. Activity类中的getPrefereneces()方法,参数为操作模式,自动使用当前活动的类名作为SharedPreferences的文件名
  3. PreferenceManager类中的getDefaultSharedPreferences()方法,接收Context参数,自动使用当前应用程序包名作为前缀来命名SharedPreferences文件

存储数据步骤:

  1. 调用SharedPreferences对象的edit()方法获取一个SharedPreferences.Editor对象

  2. 向SharedPreferences.Editor对象中添加数据,使用putBoolean、putString方法

  3. 调用apply()方法将添加的数据提交,从而完成数据存储操作

     SharedPreferences.Editor editor = getSharedPreferences("data",MODE_PRIVATE).edit();
     editor.putString("name","Tom");
     editor.putInt("age",28);
     editor.putBoolean("married",false);
     editor.apply();
    

从SharedPreferences文件中读取数据

     SharedPreferences preferences = getSharedPreferences("data",MODE_PRIVATE);
     //get方法第二个参数是默认值,当根据传入的键找不到value时,返回这个默认值
    String name = preferences.getString("name","");
    int age = preferences.getInt("age",0);
    boolean married = preferences.getBoolean("married",false);

三: SQLite数据库存储

默认存储路径:/data/data/<PackageName>/databases
数据类型:integer 整形; real 浮点型;text 文本类型; blob 二进制类型

使用SQLiteOpenHelper帮助类来方便管理数据库。而SQLiteOpenHelper是一个抽象类。我们需要自己创建一个类继承它。必须重写SQLiteOpenHelper中的两个抽象方法,onCreate()和onUpgrade(),在这两个方法中去实现创建表、升级数据库的逻辑。
SQLiteOpenHelper中有两个重要的实例方法:getReadableDatabase()和getWritableDatabase(),用于创建或打开一个现有的数据库,并返回一个可对数据库进行读或写操作的对象。

public class MyDatabaseHelper extends SQLiteOpenHelper {
     //常量化执行的sql语句
    public static final String CREATE_BOOK = "create table Book ("
            + "id integer primary key autoincrement, "
            + "author text, "
            + "price real, "
            + "pages integer, "
            + "name text)";

    public static final String CREATE_CATEGORY = "create table Category ("
            + "id integer primary key autoincrement, "
            + "category_name text, "
            + "category_code integer)";

    private Context mContext;

    /*SQLiteOpenHelper构造方法:
    参数1:context 上下文
    参数2:name 数据库名
    参数3:factory 允许我们在查询数据的时候返回一个自定义的Cursor,一般传入null
    参数4:version 表示当前数据库的版本号,用于对数据库进行升级操作
    */
    public MyDatabaseHelper(Context context, String name, SQLiteDatabase.CursorFactory factory, int version){
        super(context,name,factory,version);
        mContext = context;
    }

    @Override
    public void onCreate(SQLiteDatabase db) {
        //创建数据表
        db.execSQL(CREATE_BOOK);
        db.execSQL(CREATE_CATEGORY);
    }

    @Override
    public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {
        db.execSQL("drop table if exists Book");
        onCreate(db);
    }
}

在MainActivity中:

//实例化MyDatabaseHelper类,指定上下文、数据库名称、版本号
dbHelper = new MyDatabaseHelper(this,"BookStore.db",null,3);
dbHelper.getWritableDatabase();//检测到没有BookStore这个数据库,会创建该数据库并调用MyDatabaseHelper中的onCreated方法。

升级数据库

//当打开数据库时传入的版本号与当前的版本号不同时会调用该方法
 @Override
public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {
    //先清理数据库中所有数据表
    db.execSQL("drop table if exists Book");
    //再调用生成数据表
    onCreate(db);
}

//新添加表则应该添加在onCreate()方法中
@Override
public void onCreate(SQLiteDatabase db) {
    db.execSQL(CREATE_BOOK);
    //新添加Category表的sql语句
    db.execSQL(CREATE_CATEGORY);
}

在MainActivity中只需将Version改为大于原来版本号即可

  dbHelper = new MyDatabaseHelper(this,"BookStore.db",null,4);
  //version改为比上次大就可以触发onUpgrade方法
  dbHelper.getWritableDatabase();

向数据库添加数据

insert()方法,参数一表名,参数二是在未指定添加数据的情况下给某些可为空的列自动赋值为NULL,设置为null即可,参数三是ContentValues对象。

SQLiteDatabase db = helper.getWritableDatabase();
ContentValues values = new ContentValues();
values.put("name","The Book Name");
values.put("author","chen");
values.put("pages",100);
values.put("price",200);
db.insert("Book",null,values);

更新数据库中的数据

update()方法,参数一是表名,参数二是ContentValues对象,参数三、四是去约束更新某一行或某几行的数据,不指定默认更新所有。

SQLiteDatabase db = dbHelper.getWritableDatabase();
ContentValues values = new ContentValues();
values.put("price",10.99);

db.update("Book",values,"name = ?",new String[]{"The Da Vinci Code"});

从数据库中删除数据

delete()方法,参数一是表名,参数二、三是去约束删除某一行或某几行的数据,不指定默认删除所有。

SQLiteDatabase db = helper.getWritableDatabase();
db.delete("Book","pages> ?",new String[]{"100"});  

查询数据库中的数据

query()方法,参数一是表名,参数二是指定查询哪几列,默认全部,参数三、四是去约束查询某一行或某几行的数据,不指定默认查询所有,参数五是用于指定需要去group by的列,参数六是对group by的数据进一步的过滤,参数七是查询结果的排序方式

SQLiteDatabase db = helper.getWritableDatabase();
//query()方法返回Cursor对象
Cursor cursor = db.query("Book",null,null,null,null,null,null);
//调用cursor的moveToFirst方法,将数据的指针移到第一行的位置
if(cursor.moveToFirst()){
      do{
            //通过Cursor的getColumnIndex方法获取到某一列在列表中对应的位置索引
            String name = cursor.getString(cursor.getColumnIndex("name");
            String author = cursor.getString(cursor.getColumnIndex("author");
            int pages = cursor.getString(cursor.getColumnIndex("pages");
            double price = cursor.getString(cursor.getColumnIndex("price");
       }while(cursor.moveToNext());
}
//千万要注意,调用close()方法来关闭Cursor!!!
cursor.close():

直接使用SQL语句操作数据库

//添加数据
db.execSQL("insert into Book(name,author,pages,price) values(?,?,?,?) "
            ,new String[]{"The Book Name","chen",100,20});
//更新数据
db.execSQL("update Book set price = ? where name = ?",new String[]
            {"10","The Book Name"});
//删除数据
db.execSQL("delete from Book where pages > ?",new String[]{"100"});
//查询数据
db.rawQuery("select * from Book",null);

查看模拟器中应用数据: adb shell命令,然后cd到data/data/包名/databases下查看

若adb shell 里面 ls 提示opendir failed, Permission denied 没有权限,则只需要adb root,adb remount 一下就可以了

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,904评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,581评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,527评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,463评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,546评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,572评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,582评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,330评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,776评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,087评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,257评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,923评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,571评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,192评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,436评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,145评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,127评论 2 352

推荐阅读更多精彩内容

  • 任何一个应用程序,其实说白了就是在不停地和数据打交道,我们聊QQ、看新闻、刷微博,所关心的都是里面的数据,...
    AndYMJ阅读 1,652评论 2 5
  • 1.引言   一般在做一些面试题的时候,Android有几种数据存储方案这个问题是经常碰到的。在我们实际应用中,任...
    忆念成风阅读 2,136评论 0 10
  • 我问小草小草摇摇摇出你的身腰 我问小河小河漾漾漾出你的甜笑 我问云儿云儿幻幻幻出你的愁容 我问燕子燕子摆摆不理不睬...
    Queequeg阅读 428评论 0 2
  • 许桑抿了抿嘴唇“嗯,是朋友。”“是男朋友吗?”这一问如平地一声雷,许桑顿时就毛了“跟你有什么关系啊?”这人简直是无...
    忆夕笑雪阅读 308评论 0 0
  • 是多久没写过日志,以致不知该从哪写起。所以也就顺意起头,行文,走起 今年算是挺有感触的一年,工作从早晚颠倒的行业华...
    吸鼻涕阅读 331评论 0 0