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

本文实例讲述了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程序设计有所帮助。

时间: 2024-12-28 15:58:20

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

Android 中 onSaveInstanceState()使用方法详解

Android 中 onSaveInstanceState()使用方法详解 覆盖onSaveInstanceState方法,并在onCreate中检测savedInstanceState和获取保存的值 @Override protected void onSaveInstanceState(Bundle outState) { outState.putInt("currentposition", videoView.getCurrentPosition()); super.onSave

Ruby中的钩子方法详解

  这篇文章主要介绍了Ruby中的钩子方法详解,本文讲解了什么是钩子方法.included.Devise中的 included.extended.ActiveRecord中的 extended.prepended.inherited等内容,需要的朋友可以参考下 Ruby的哲学理念是基于一个基本的要素,那就是让程序员快乐.Ruby非常注重程序员的快乐,并且也提供了许多不同的方法来实现它. 它的元编程能力能够让程序员编写在运行时动态生成的代码.它的线程功能使得程序员有一种优雅的的方式编写多线程代码.

Android中HorizontalScrollView使用方法详解_Android

由于移动设备物理显示空间一般有限,不可能一次性的把所有要显示的内容都显示在屏幕上.所以各大平台一般会提供一些可滚动的视图来向用户展示数据.Android平台框架中为我们提供了诸如ListView.GirdView.ScrollView等滚动视图控件,这几个视图控件也是我们平常使用最多的.下面介绍一下HorizontalScrollView的使用和需要注意的点:  HorizontalScrollView是一个FrameLayout  ,这意味着你只能在它下面放置一个子控件,这个子控件可以包含很多

Android编程实现XML解析与保存的三种方法详解

本文实例讲述了Android编程实现XML解析与保存的三种方法.分享给大家供大家参考,具体如下: 简介 在Android开发中,关于XML解析有三种方式,分别是: 1. SAX 基于事件的解析器,解析速度快,占用内存少.非常适合在Android移动设备中使用. 2. DOM 在内存中以树形结构存放,因此检索和更新效率会更高.但是对于特别大的文档,解析和加载整个文档将会很耗资源 3. PULL 基于事件的解析器,不同于SAX是,PULL是主动请求下一个事件,所以在可控上PULL要比SAX实用.An

ThinkPHP中where()使用方法详解_php实例

本文介绍ThinkPHP的where()方法的用法.where方法可以用于对数据库操作的结果进行筛选.即SQL查询语句中的where子句. 今天来给大家讲下查询最常用但也是最复杂的where方法,where方法也属于模型类的连贯操作方法之一,主要用于查询和操作条件的设置. where方法的用法是ThinkPHP查询语言的精髓,也是ThinkPHP ORM的重要组成部分和亮点所在,可以完成包括普通查询.表达式查询.快捷查询.区间查询.组合查询在内的查询操作.where方法的参数支持字符串和数组,虽

ThinkPHP中limit()使用方法详解_php实例

本文介绍ThinkPHP的limit()方法的用法.limit方法可以用于对数据库操作的结果进行取指定范围的条数.即相当于是在mysql查询语句中的limit子句. limit方法也是模型类的连贯操作方法之一,主要用于指定查询和操作的数量,特别在分页查询的时候使用较多.ThinkPHP的l imit方法可以兼容所有的数据库驱动类的. 用法一.限制结果数量 例如获取满足要求的10个用户,如下调用即可: $User = M('User'); $User->where('status=1')->fi

深入PHP5中的魔术方法详解_php技巧

从PHP 5以后的版本,PHP中的类就可以使用魔术方法了.其规定以两个下划线(__)开头的方法都保留为魔术方法,所以建议大家函数名最好不用__开头,除非是为了重载已有的魔术方法. 1.__get.__set这两个方法是为在类和他们的父类中没有声明的属性而设计的.◆__get( $property ) 当调用一个未定义的属性时,此方法会被触发,传递的参数是被访问的属性名.◆__set( $property, $value ) 给一个未定义的属性赋值时,此方法会被触发,传递的参数是被设置的属性名和值

Android studio利用gradle打jar包并混淆的方法详解

本文主要介绍了Android studio利用gradle打jar包并混淆的方法,下面话不多说,来看看详细的介绍吧. 首先打jar包的配置很简单,使用jar的task,可以参考gradle官方文档,具体代码如下: task buildJar(type: Jar, dependsOn: ['assembleRelease']) { destinationDir = file('build/outputs/jar/') appendix = "" baseName = "&quo

Android编程之ICS式下拉菜单PopupWindow实现方法详解(附源码下载)_Android

本文实例讲述了Android编程之ICS式下拉菜单PopupWindow实现方法.分享给大家供大家参考,具体如下: 运行效果截图如下: 右边这个就是下拉菜单啦,看见有的地方叫他 ICS式下拉菜单,哎哟,不错哦! 下面先讲一下实现原理: 这种菜单实际上就是一个弹出式的菜单,于是我们想到android PopupWindow 类,给他设置一个view 在弹出来不就OK了吗. PopupWindow 的用法也很简单 主要方法: 步骤1.new 一个实例出来,我们使用这个构造方法即可, 复制代码 代码如