Android数据持久化之读写SD卡中内容的方法详解
发布时间 - 2026-01-11 00:55:52 点击率:次本文实例讲述了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文件添加如下代码:
<!-- 获取写权限 --> <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
那么这个文件保存到哪了呢?在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文件:
<?xml version="1.0" encoding="utf-8"?>
<products>
<product>
<id>1</id>
<name>电脑</name>
<price>3088</price>
</product>
<product>
<id>2</id>
<name>微波炉</name>
<price>2500</price>
</product>
<product>
<id>3</id>
<name>洗衣机</name>
<price>1088</price>
</product>
</products>
定义一个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<Product> products;
Product product;
StringBuffer sb = new StringBuffer();
public List<Product> getProduct() {
return products;
}
/**
* 开始分析XML文件
*/
@Override
public void startDocument() throws SAXException {
// 开始分析XML文件,创建list对象用于保存分析完的product对象
products = new ArrayList<Product>();
super.startDocument();
}
/**
* 开始分析XML中的标签
*/
@Override
public void startElement(String uri, String localName, String qName,
Attributes attributes) throws SAXException {
if (localName.equals("product")) {
// 如果开始分析的是<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>标签后,将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<Product> 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相关内容感兴趣的读者可查看本站专题:《Android编程开发之SD卡操作方法汇总》、《Android文件操作技巧汇总》、《Android数据库操作技巧总结》、《Android编程之activity操作技巧总结》、《Android开发入门与进阶教程》、《Android资源操作技巧汇总》、《Android视图View技巧总结》及《Android控件用法总结》
希望本文所述对大家Android程序设计有所帮助。
# Android
# 数据持久化
# 读写SD卡
# Android SQLite操作之大数据处理与同时读写方法
# Android通过json向MySQL中读写数据的方法详解【读取篇】
# Android通过json向MySQL中读写数据的方法详解【写入篇】
# Android实现读写JSON数据的方法
# android读写sd卡操作写入数据读取数据示例
# Android实现读写USB串口数据
# 目录下
# 的是
# 操作技巧
# 子类
# 创建一个
# 卡上
# 进阶
# 相关内容
# 最多
# 在这
# 这就是
# 都能
# 最重要
# 中有
# 目录中
# 但在
# 这一步
# 将其
# 感兴趣
# 能在
相关栏目:
【
网站优化151355 】
【
网络推广146373 】
【
网络技术251813 】
【
AI营销90571 】
相关推荐:
jQuery 常见小例汇总
邀请函制作网站有哪些,有没有做年会邀请函的网站啊?在线制作,模板很多的那种?
Laravel队列任务超时怎么办_Laravel Queue Timeout设置详解
Laravel怎么配置自定义表前缀_Laravel数据库迁移与Eloquent表名映射【步骤】
新三国志曹操传主线渭水交兵攻略
HTML5空格和margin有啥区别_空格与外边距的使用场景【说明】
如何在自有机房高效搭建专业网站?
Java Adapter 适配器模式(类适配器,对象适配器)优缺点对比
浅谈javascript alert和confirm的美化
矢量图网站制作软件,用千图网的一张矢量图做公司app首页,该网站并未说明版权等问题,这样做算不算侵权?应该如何解决?
Swift中swift中的switch 语句
高性能网站服务器配置指南:安全稳定与高效建站核心方案
怎样使用JSON进行数据交换_它有什么限制
Laravel如何处理CORS跨域请求?(配置示例)
Laravel怎么定时执行任务_Laravel任务调度器Schedule配置与Cron设置【教程】
如何使用 jQuery 正确渲染 Instagram 风格的标签列表
Java垃圾回收器的方法和原理总结
如何正确选择百度移动适配建站域名?
JavaScript实现Fly Bird小游戏
Laravel如何实现多对多模型关联?(Eloquent教程)
🚀拖拽式CMS建站能否实现高效与个性化并存?
如何确保FTP站点访问权限与数据传输安全?
大型企业网站制作流程,做网站需要注册公司吗?
JavaScript如何操作视频_媒体API怎么控制播放
手机钓鱼网站怎么制作视频,怎样拦截钓鱼网站。怎么办?
高端建站如何打造兼具美学与转化的品牌官网?
如何彻底卸载建站之星软件?
JavaScript如何实现音频处理_Web Audio API如何工作?
Midjourney怎么调整光影效果_Midjourney光影调整方法【指南】
Laravel如何配置和使用缓存?(Redis代码示例)
清除minerd进程的简单方法
WEB开发之注册页面验证码倒计时代码的实现
Laravel如何使用缓存系统提升性能_Laravel缓存驱动和应用优化方案
百度浏览器如何管理插件 百度浏览器插件管理方法
如何获取上海专业网站定制建站电话?
Laravel如何升级到最新的版本_Laravel版本升级流程与兼容性处理
Laravel如何连接多个数据库_Laravel多数据库连接配置与切换教程
如何确认建站备案号应放置的具体位置?
Laravel如何集成Inertia.js与Vue/React?(安装配置)
Laravel如何实现API版本控制_Laravel API版本化路由设计策略
如何获取PHP WAP自助建站系统源码?
北京的网站制作公司有哪些,哪个视频网站最好?
jQuery validate插件功能与用法详解
JavaScript如何实现路由_前端路由原理是什么
Laravel如何使用Blade模板引擎?(完整语法和示例)
香港网站服务器数量如何影响SEO优化效果?
C++时间戳转换成日期时间的步骤和示例代码
韩国代理服务器如何选?解析IP设置技巧与跨境访问优化指南
安克发布新款氮化镓充电宝:体积缩小 30%,支持 200W 输出
laravel服务容器和依赖注入怎么理解_laravel服务容器与依赖注入解析

