一、文件存储

1.1.将数据存储到文件中

1.运用java的io流将数据写入到手机文件中

/**
 * 保存数据到文件
 * */
public void save(String data){
    FileOutputStream out = null;
    BufferedWriter writer = null;
    try {
        //第一个参数为文件名,第二个参数为操作模式
        out = openFileOutput("data", Context.MODE_PRIVATE);
        //将字节流包装成字符流
        writer = new BufferedWriter(new OutputStreamWriter(out));
        //写入数据
        writer.write(data);
    } catch (IOException e) {
        e.printStackTrace();
    } finally {
            try {
                if(writer!=null) {
                    //结束的时候关闭流
                    writer.close();
                }
            } catch (IOException e) {
                e.printStackTrace();
            }
        }
    }

在Context类中,提供了一个openFileOutput()方法,可以用于获取一个FileOutputStream的字节流对象,从而实现将数据存储到手机指定文件中的功能。其接收的两个参数中,第一个参数为文件名,(所有文件都默认存储到data/data/<package name>/files/目录下)第二个参数为操作模式,示例中的MODE_PRIVATE是默认操作模式,表示当指定同样文件名时,所写的内容将会覆盖原内容。常用的另一种模式是MODE_APPEND,它表示如果该文件存在,就在文件里面追加内容,不存在就创建新文件。

1.2.从文件中读取数据

1.运用java的io流将数据从手机文件中读取出来

/**
 * 从文件中读取数据
 * */
public String load(){
    FileInputStream in = null;
    BufferedReader reader = null;
    StringBuilder content = new StringBuilder();
    try {
        //参数为文件名
        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();
}

类似的,在Context类中,还提供了一个openFileInput()方法,可以用于获取一个FileInputStream的字节流对象,从而实现从手机文件中读取数据的功能。这里只接收一个参数,即文件名。

二、SharedPreferences存储

2.1.将数据存储到SharedPreferences中

1.获取SharedPreferences对象

Android中主要提供了三种方法用于得到SharedPreferences对象:

a.Context类中的getSharedPreferences()方法

getBaseContext().getSharedPreferences("data",MODE_PRIVATE);

此方法接收两个参数,第一个参数为SharedPreferences文件的名称,(文件默认存储到data/data/<package name>/shared_prefs/目录下)第二个参数为操作模式,目前仅MODE_PRIVATE这一种模式可选,为默认操作模式,表示只有当前应用程序才可以对这个SharedPreferences文件进行读写。

b.Activity类中的getPreferences()方法

getPreferences(MODE_PRIVATE);

此方法和Context类中的getSharedPreferences()方法相似,不过它只接收一个操作模式参数,因为它会用当前活动的类名作为SharedPreferences的文件名。

c.PreferenceManager类中的getDefaultSharedPreferences()方法

PreferenceManager.getDefaultSharedPreferences(getBaseContext());

该方法接收一个Context参数,并自动使用当前应用程序的包名作为前缀来命名SharedPreferences文件。

2.获取SharedPreferences.Editor对象

SharedPreferences.Editor editor = getSharedPreferences("data",MODE_PRIVATE).edit();

3.向Editor中添加数据(键值对的形式)

editor.putString("name","Ein");

4.提交数据,完成存储操作

editor.apply();

PS:editor的提交数据除了apply()还有commit()

这两个方法的区别在于: 
1. apply没有返回值而commit返回boolean表明修改是否提交成功 
2. apply是将修改数据原子提交到内存, 而后异步真正提交到硬件磁盘, 而commit是同步的提交到硬件磁盘,因此,在多个并发的提交commit的时候,他们会等待正在处理的commit保存到磁盘后在操作,从而降低了效率。而apply只是原子的提交到内容,后面有调用apply的函数的将会直接覆盖前面的内存数据,这样从一定程度上提高了很多效率。 
3. apply方法不会提示任何失败的提示。 
由于在一个进程中,sharedPreference是单实例,一般不会出现并发冲突,如果对提交的结果不关心的话,建议使用apply,当然需要确保提交成功且有后续操作的话,还是需要用commit的。 

2.2.从SharedPreferences中读取数据

1.获取SharedPreferences对象(这里就仅以Context类中的getSharedPreferences()方法为例)

SharedPreferences preferences = getSharedPreferences("data",MODE_PRIVATE);

2.根据键去读取值,这里接收两个参数,第一个参数为键,第二个参数为缺省值

String name = preferences.getString("name","");

三、数据库存储(使用LitePal)

3.1.配置LitePal

1.在app/build.gradle文件的dependencies闭包中添加开源库LitePal

implementation 'org.litepal.android:core:2.0.0'

2.配置litepal.xml文件

在app/src/main/assets目录下新建一个litepal.xml文件,并编辑

<?xml version="1.0" encoding="utf-8" ?>
<litepal>
    <dbname value="BookStore"></dbname>
    <version value="1"></version>
    <list>
        
    </list>
</litepal>

其中<dbname>标签用于指定数据库名称,<version>标签用于指定数据库版本号,<list>标签用于指定所有的映射模型。

3.编辑AndroidManifest.xml文件

修改<application>标签的name属性值为"org.litepal.LitePalApplication"

<application
    android:name="org.litepal.LitePalApplication"
    android:allowBackup="true"
    android:icon="@mipmap/ic_launcher"
    android:label="@string/app_name"

如果<application> 便签中的name属性已经配置了值,那么我们可以在Application中调用LitePal.initialize(context)方法起到同样的效果。

public class MyOwnApplication extends Application {

    @Override
    public void onCreate() {
        super.onCreate();
        LitePal.initialize(this);
    }
    ...
}

3.2.创建和升级数据库

1.新建一个Java bean(这里以Book类为例),类对应于数据库中的表,类中每一个成员变量对应于表中的每一个列。

public class Book {
    private int id;
    private String author;
    private double price;
    private int pages;
    private String name;
...
}

 PS:这里即使不声明private int id这条成员变量,创建数据库的时候仍会为Book表新增主键id。但是需要getId()方法才能获取。

关于注释的补充说明:

//@Column(defaultValue = "unknown")设置缺省值。
//@Column (unique = true) 主键
//@Column (ignore = true) 忽略,创建数据库不会添加该键

2.修改litepal.xml文件,将Book类添加到映射模型列表当中(配置值为其完整类名)

<list>
    <mapping class="com.xxxx.xxxxx.litepaltest.Book"></mapping>
</list>

3.调用创建数据库的方法

Connector.getDatabase();

 4.升级数据库的时候,如果修改表,则直接在Java bean类中进行修改,比如这里在Book表中添加一个出版社的列:

private int id;
private String author;
private double price;
private int pages;
private String name;
private String press;

如果要新增表,则再新建一个Java bean类与之对应,比如这里新建一张Category表格:

public class Category {
    private int id;
    private String categoryName;
    private int categoryCode;
...
}

同样的,每新建一张表格,都需要将其添加到映射模型列表当中:

<list>
    <mapping class="com.xxxx.xxxxx.litepaltest.Book"></mapping>
    <mapping class="com.xxxx.xxxxx.litepaltest.Category"></mapping>
</list>

最后让版本号加1:

<version value="2"></version>

再重新调用getDatabase()方法时,便会升级数据库了。

Connector.getDatabase();

3.3.使用LitePal添加数据

1.让Java bean类继承 LitePalSupport类

public class Book extends LitePalSupport
public class Category extends LitePalSupport

2.新建Book实例,通过save()方法保存数据

Book book = new Book();
book.setName("The Da Vinci Code");
book.setAuthor("Dan Brown");
book.setPages(454);
book.setPrice(16.96);
book.setPress("Unknow");
book.save();

3.4.使用LitePal更新数据

1.最简单的方法就是通过find()方法找到对应条目,并通过save()方法更新数据。

//第一个参数为查询哪张表,第二个参数为查询第几条数据
Book book = LitePal.find(Book.class,1);
book.setPrice(20.99);
book.save();

2.或者使用update()方法更新数据

Book book = new Book();
book.setName("The Lost Symbol");
book.update(id);

3.或者使用updateAll()方法批量更新数据

Book book = new Book();
book.setPress("Anchor");
book.setPages(576);
book.updateAll("name = ? and author = ?","The Lost Symbol","Dan Brown");

PS:当我们想将某字段更新成缺省值时,我们应该使用setToDefault()方法:

Book book = new Book();
book.setToDefault("pages");
book.updateAll();

3.5.使用LitePal删除数据

1.根据主键删除

LitePal.delete(Book.class, id);

2.批量删除

LitePal.deleteAll(Book.class, "price > ?" , "21");

3.6.使用LitePal查询数据

1.主键查询

Book book = LitePal.find(Book.class, id);

2.查询所有数据

List<Book> allBooks = LitePal.findAll(Book.class);

3.条件查询

List<Book> books = LitePal.select("name","author","pages")
        .where("pages > ?","400")
        .order("pages")
        .limit(10)
        .offset(10)
        .find(Book.class);

其中,select用于指定查询哪几列的数据;where用于指定查询的约束条件;order用于指定排序方式,desc表示降序排序(例如pages desc),asc或者不写表示升序排列;limit用于指定查询结果的数量;offset用于指定查询结果的偏移量。