Android数据持久化之读写SD卡中内容的方法详解

发布时间 - 2026-01-11 00:55:52    点击率:

本文实例讲述了Android数据持久化之读写SD卡中内容的方法。分享给大家供大家参考,具体如下:

前面文章里讲的那三个方法:openFileOutput openFileInput 虽然都能通过流对象OutputStreamInputStream可以处理任意文件中的数据,但与 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服务容器与依赖注入解析