本文实例讲述了Android数据持久化之读写SD卡中内容的方法。分享给大家供大家参考,具体如下:
前面文章里讲的那三个方法:openFileOutput、openFileInput虽然都能通过流对象OutputStream和InputStream可以处理任意文件中的数据,但与 SharedPreferences 一样,只能在手机内存的指定目录下建立文件,因此,在实际的开发使用中有很大的局限性,那么在这一节中,我们来看一个比较高级的方法来实现数据的持久化——读写SD卡上的内容。
——读取assets目录中的文件
android中的文件夹assets存放的是二进制的文件格式,比如音频、视频、图片等,但该目录下的文件不会被R.java文件索引到,如果想读取该目录下的文件还需要借助AssetManager对象。
代码如下:
/**
* 将图片文件保存到SD卡的根目录下
*
* 虽然确定SD卡的路径是可以直接使用"/sdcard"的,但在实际开发中建议使用:android.os.Environment.getExternalStorageDirectory()
* 方法获得SD卡的路径,这样一旦系统改变了路径,应用程序会立刻获得最新的SD卡的路径,这样做会使程序更健壮。
*/
public void writeToSD() {
try {
//创建用于将图片保存到SD卡上的FileOutputStream对象
FileOutputStream fos = new FileOutputStream(android.os.Environment.getExternalStorageDirectory() + "/image.jpg");
//打开assets目录下的image.jpg文件,并返回InputStream对象
InputStream is = getResources().getAssets().open("image.jpg");
//定义一个byte数组,用来保存每次向SD卡中文件写入的数据,最多8k
byte[] buffer = new byte[8192];
int count = 0;
//循环写入数据
while((count = is.read(buffer)) != -1)
{
fos.write(buffer, 0, count);
}
fos.close();
is.close();
Toast.makeText(this, "已成功将图片保存在SD卡中", Toast.LENGTH_SHORT).show();
} catch (Exception e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
/**
* 从SD卡中读取图片文件
* @throws IOException
*/
public void readFromSD() throws IOException{
//指定SD卡中的图像文件名
String fileName = android.os.Environment.getExternalStorageState() + "image.jpg";
//判断文件图片是否存在
if (!new File(fileName).exists()) {
Toast.makeText(this, "没有要找的图片文件,未装入", Toast.LENGTH_SHORT).show();
return;
}
image = (ImageView) findViewById(R.id.image);
FileInputStream fis = new FileInputStream(fileName);
//从文件的输入流装载Bimap对象
Bitmap bitmap = BitmapFactory.decodeStream(fis);
image.setImageBitmap(bitmap);
fis.close();
}
从android2.x开始,默认不允许向SD卡中写文件,因此要添加权限,在AndroidManifest.xml文件添加如下代码:
那么这个文件保存到哪了呢?在Eclipse中进入File Explorer 面板,选中/data/app目录下的该程序的APK文件,将其导出到桌面上或者其他地方,解压后进入assets目录可看见刚才保存的图片。
由于assets文件夹下的文件是被打包进apk文件中的,所以assets目录中的文件只能读,不能写。
——SAX引擎读取XML文件
原理:
android SDK 本身提供了操作XML文件的类库,这就是SAX,使用SAX处理XML需要一个Handler对象,一般会使用:org.xml.sax.helpers.DefaultHandler 的子类来创建Handler对象。SAX技术处理XML文件时并不是一次性的把XML文件装入内存,而是一边读一边解析,因此,就需要如下的五个分析点(分析事件):
1、开始分析XML文件:对应方法 DefaultHandler.startDocument 可以在该方法中做一些初始化的工作
2、开始处理每一个XML标签,即每个标签对的起始标签:对应方法 startElement 该方法可以获取当前标签的名称、属性的相关信息
3、处理完每一个XML标签,即每个标签对的结束标签:对应方法 endElement 获得当前处理的标签的全部信息
4、处理完XML文件,即处理完了整个XML文件的内容时,就到这一步了,对应方法:endDocument
5、读取字符分析点,是对上述获取到的XML文件的全部内容的处理,这一步很重要,对应方法:characters 用来处理获取到的XML文件中的内容,即保存XML标签中的内容。
如下是对上面五点的应用,将XML文件转换成java对象:
首先在/res/raw 下创建一个wxml文件:
1
电脑
3088
2
微波炉
2500
3
洗衣机
1088
定义一个product类:
package com.example.data_io_xmltojava;
public class Product {
int id;
String name;
int price;
public int getId() {
return id;
}
public void setId(int id) {
this.id = id;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getPrice() {
return price;
}
public void setPrice(int price) {
this.price = price;
}
}
下面是XML2Product类,是DefaultHandler的子类,这个类是整个程序中最重要最核心的类:
package com.example.data_io_xmltojava;
import java.util.ArrayList;
import java.util.List;
import org.xml.sax.Attributes;
import org.xml.sax.SAXException;
import org.xml.sax.helpers.DefaultHandler;
public class XML2Product extends DefaultHandler {
List products;
Product product;
StringBuffer sb = new StringBuffer();
public List getProduct() {
return products;
}
/**
* 开始分析XML文件
*/
@Override
public void startDocument() throws SAXException {
// 开始分析XML文件,创建list对象用于保存分析完的product对象
products = new ArrayList();
super.startDocument();
}
/**
* 开始分析XML中的标签
*/
@Override
public void startElement(String uri, String localName, String qName,
Attributes attributes) throws SAXException {
if (localName.equals("product")) {
// 如果开始分析的是标签,创建一个product对象
product = new Product();
}
super.startElement(uri, localName, qName, attributes);
}
/**
* 分析完了XML中的标签
* 使用sb中的值为product对象中的属性赋值
*/
@Override
public void endElement(String uri, String localName, String qName)
throws SAXException {
if (localName.equals("product")) {
// 处理完标签后,将product对象添加到products中
products.add(product);
} else if (localName.equals("id")) {
// 设置id属性值
product.setId(Integer.parseInt(sb.toString().trim()));
// 将保存标签内容的缓存区清空
sb.setLength(0);
} else if (localName.equals("name")) {
product.setName(sb.toString().trim());
sb.setLength(0);
} else if (localName.equals("price")) {
product.setPrice(Integer.parseInt(sb.toString().trim()));
sb.setLength(0);
}
super.endElement(uri, localName, qName);
}
/**
* 分析完了XML文件
*/
@Override
public void endDocument() throws SAXException {
super.endDocument();
}
/**
* 处理SAX读取到的XML文件中的内容
*/
@Override
public void characters(char[] ch, int start, int length)
throws SAXException {
// 将SAX扫描到的内容保存到sb变量中
sb.append(ch, start, length);
super.characters(ch, start, length);
}
}
下面的就是将xml文件转化成java对象的类了:
package com.example.data_io_xmltojava;
import java.io.IOException;
import java.io.InputStream;
import java.util.List;
import org.xml.sax.SAXException;
import android.os.Bundle;
import android.app.Activity;
import android.app.AlertDialog;
import android.util.Xml;
import android.view.Menu;
public class MainActivity extends Activity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
// 获得 /res/raw/products.xml文件中InputStream对象
InputStream is = getResources().openRawResource(R.raw.products);
XML2Product xml2product = new XML2Product();
try {
// 开始分析products.xml文件(解析)
android.util.Xml.parse(is, Xml.Encoding.UTF_8, xml2product);
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (SAXException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
// 将转换后得到的java对象的内容输出
List products = xml2product.getProduct();
String msg = "total" + products.size() + "\n";
for (Product product : products) {
msg += "id:" + product.getId() + "产品名:" + product.getName() + "价格"
+ product.getPrice() + "\n";
}
new AlertDialog.Builder(this).setTitle("产品信息").setMessage(msg)
.setPositiveButton("关闭", null).show();
}
@Override
public boolean onCreateOptionsMenu(Menu menu) {
// Inflate the menu; this adds items to the action bar if it is present.
getMenuInflater().inflate(R.menu.main, menu);
return true;
}
}
希望本文所述对大家Android程序设计有所帮助。