Android核心基础-5.Android 数据存储与访问-1.使用文件进行数据存储

五种数据存储方式
很多时候我们的软件需要对处理后的数据进行存储或再次访问。Android为数据存储提供了如下五种方式:

  1. 文件
  2. SharedPreferences(参数)
  3. SQLite 数据库
  4. Content provider 内容提供者
  5. 网络

一、使用文件进行数据存储

1.1 写内容到文件-openFileOutput()简介

在上下文中有一个方法叫openFileOutput()方法可以用于把数据输出到文件中,具体的实现过程与在J2SE环境中保存数据到文件中是一样的。
FileOutputStream outStream = this.openFileOutput(“student.txt”, Context.MODE_PRIVATE);
outStream.write(“深情小建”.getBytes());
outStream.close();
openFileOutput()方法的第一参数用于指定文件名称,不能包含路径分隔符“/”,如果文件不存在,Android会自动创建它。创建的文件保存在/data/data/< package name>/files目录,如:/data/data/net.dxs/files/student.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

  1. Context.MODE_PRIVATE:
    为默认操作模式,代表该文件是私有数据,只能被应用本身访问,在该模式下,写入的内容会覆盖原文件的内容,如果想把新写入的内容追加到原文件中。可以使用Context.MODE_APPEND

  2. Context.MODE_APPEND:
    模式会检查文件是否存在,存在就往文件追加内容,否则就创建新文件。

Context.MODE_WORLD_READABLE和Context.MODE_WORLD_WRITEABLE用来控制其他应用是否有权限读写该文件。

  1. MODE_WORLD_READABLE:表示当前文件可以被其他应用读取;
  2. MODE_WORLD_WRITEABLE:表示当前文件可以被其他应用写入。
    如果希望文件被其他应用读和写,可以传入:
    openFileOutput(“student.txt”, Context.MODE_WORLD_READABLE + Context.MODE_WORLD_WRITEABLE);

android有一套自己的安全模型,当应用程序(.apk)在安装时系统就会分配给他一个userid,当该应用要去访问其他资源比如文件的时候,就需要userid匹配。默认情况下,任何应用创建的文件,sharedpreferences,数据库都应该是私有的(位于/data/data/< package name>/files),其他程序无法访问。除非在创建时指定了Context.MODE_WORLD_READABLE或者Context.MODE_WORLD_WRITEABLE ,只有这样其他程序才能正确访问。

1.2 读取文件内容-openFileInput()简介

如果要打开存放在/data/data/< package name>/files目录应用私有的文件,可以使用Activity提供openFileInput()方法。
FileInputStream inStream = this.getContext().openFileInput(“student.txt”);
Log.i(“FileTest”, readInStream(inStream));
readInStream()的方法如下:

public static String readInStream(FileInputStream inStream){
    try {
        ByteArrayOutputStream outStream = new ByteArrayOutputStream();
        byte[] buffer = new byte[1024];
        int length = -1;
        while((length = inStream.read(buffer)) != -1 ){
            outStream.write(buffer, 0, length);
        }
        outStream.close();
        inStream.close();
        return outStream.toString();
    } catch (IOException e) {
        Log.i("FileTest", e.getMessage());
    }
    return null;
}

或者直接使用文件的绝对路径:
File file = new File(“/data/data/net.dxs/files/student.txt”);
FileInputStream inStream = new FileInputStream(file);
Log.i(“FileTest”, readInStream(inStream));
注意:上面文件路径中的“net.dxs”为应用所在包,当你在编写代码时应替换为你自己应用使用的包。
对于私有文件只能被创建该文件的应用访问,如果希望文件能被其他应用读和写,可以在创建文件时,指定Context.MODE_WORLD_READABLE和Context.MODE_WORLD_WRITEABLE权限。

Activity还提供了getCacheDir()和getFilesDir()方法:
getCacheDir()方法用于获取/data/data/< package name>/cache目录
getFilesDir()方法用于获取/data/data/< package name>/files目录

1.3 把文件存放在SDCard

使用Activity的openFileOutput()方法保存文件,文件是存放在手机空间上,一般手机的存储空间不是很大,存放些小文件还行,如果要存放像视频这样的大文件,是不可行的。对于像视频这样的大文件,我们可以把它存放在SDCard。 SDCard是干什么的?你可以把它看作是移动硬盘或U盘。

在模拟器中使用SDCard,你需要先创建一张SDCard卡(当然不是真的SDCard,只是镜像文件)。创建SDCard可以在Eclipse创建模拟器时随同创建。

在程序中访问SDCard,你需要申请访问SDCard的权限。
// 挂载和卸载SDCard
android.permission.MOUNT_UNMOUNT_FILESYSTEMS
// 写入外存储设备权限
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, “student.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, "student.txt"); 
//上面两句代码可以合成一句: File saveFile = new File("/mnt/sdcard/student.txt");
FileOutputStream outStream = new FileOutputStream(saveFile);
outStream.write("深情小建test".getBytes());
outStream.close();

1.4 使用pull解析XML文件

下面是本例子要解析的XML文件:

<?xml version='1.0' encoding='utf-8' standalone='yes' ?>
<students>
    <student id="1">
        <name>深情小建1</name>
        <sex></sex>
        <age>10</age>
    </student>
    <student id="2">
        <name>深情小建2</name>
        <sex></sex>
        <age>10</age>
    </student>
</students>

例子定义了一个java bean用于存放上面解析出来的xml内容, 这个java bean为Person,代码如下:

package net.dxs.domain;

/**
 * @author lijian
 * 
 */
public class Student {
    private Integer id;
    private String name;
    private String sex;
    private String age;

    public Student() {
    }

    public Student(Integer id, String name, String sex, String age) {
        this.id = id;
        this.name = name;
        this.sex = sex;
        this.age = age;
    }

    public Integer getId(){
        return id;
    }

    public void setId(Integer id) {
        this.id = id;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public String getSex() {
        return sex;
    }

    public void setSex(String sex) {
        this.sex = sex;
    }

    public String getAge() {
        return age;
    }

    public void setAge(String age) {
        this.age = age;
    }

    @Override
    public String toString() {
        return "Student [id=" + id + ", name=" + name + ", sex=" + sex + ", age=" + age + "]";
    }
}

有些时候,我们需要生成一个XML文件,生成XML文件的方法有很多,如:可以只使用一个StringBuilder组拼XML内容,然后把内容写入到文件中;或者使用DOM API生成XML文件,或者也可以使用pull解析器生成XML文件,这里推荐大家使用Pull解析器。

使用Pull解析器生成一个与student.xml文件内容相同的student.xml文件,代码如下:

生成Xml数据:
public static String writeXML(List<Person> persons, Writer writer){
    XmlSerializer serializer = Xml.newSerializer();
    try {
        serializer.setOutput(writer);
        serializer.startDocument("UTF-8", true);
        //第一个参数为命名空间,如果不使用命名空间,可以设置为null
        serializer.startTag("", "persons");
        for (Person person : persons){
            serializer.startTag("", "person");
            serializer.attribute("", "id", person.getId().toString());

            serializer.startTag("", "name");
            serializer.text(person.getName());
            serializer.endTag("", "name");

            serializer.startTag("", "sex");
            serializer.text(person.getSex());
            serializer.endTag("", "sex");

            serializer.startTag("", "age");
            serializer.text(person.getAge());
            serializer.endTag("", "age");

            serializer.endTag("", "person");
        }
        serializer.endTag("", "persons");
        serializer.endDocument();
        return writer.toString();
    } catch (Exception e) {
        e.printStackTrace();
    }
    return null;
}

解析Xml文件:
public static List<Person> readXML(InputStream inStream) {
    XmlPullParser parser = Xml.newPullParser();
    try {
    parser.setInput(inStream, "UTF-8");
    int eventType = parser.getEventType();
    Person currentPerson = null;
    List<Person> persons = null;
    while (eventType != XmlPullParser.END_DOCUMENT) {
        switch (eventType) {
        case XmlPullParser.START_DOCUMENT://文档开始事件,可以进行数据初始化处理
            persons = new ArrayList<Person>();
            break;
        case XmlPullParser.START_TAG://开始元素事件
            String name = parser.getName();
            if (name.equals("person")) {
                currentPerson = new Person();
                currentPerson.setId(new Integer(parser.getAttributeValue(null, "id")));
            } else if (currentPerson != null) {
                if (name.equals("name")) {
                    currentPerson.setName(parser.nextText());// 如果后面是Text节点,即返回它的值
                }else if(name.equals("sex")){
                    currentPerson.setName(parser.nextText());
                } else if (name.equals("age")) {
                    currentPerson.setAge(parser.nextText());
                }
            }
            break;
        case XmlPullParser.END_TAG://结束元素事件
            if (parser.getName().equals("person") && currentPerson != null) {
                persons.add(currentPerson);
                currentPerson = null;
            }
            break;
        }
        eventType = parser.next();
    }
    inStream.close();
    return persons;
    } catch (Exception e) {
        e.printStackTrace();
    }
    return null;
}

使用代码如下(生成XML文件):
File xmlFile = new File(“student.xml”);
FileOutputStream outStream = new FileOutputStream(xmlFile);
OutputStreamWriter outStreamWriter = new OutputStreamWriter(outStream, “UTF-8”);
BufferedWriter writer = new BufferedWriter(outStreamWriter);
writeXML(persons, writer);
writer.flush();
writer.close();
如果只想得到生成的xml字符串内容,可以使用StringWriter:
StringWriter writer = new StringWriter();
writeXML(persons, writer);
String content = writer.toString();

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值