我们在实际开发中,有的时候需要储存或者备份比较复杂的数据。这些数据的特点是,内容多、结构大,比如短信备份等。我们知道SharedPreferences和Files(文本文件)储存这种数据会非常的没有效率。如果学过JavaWeb的朋友,首先可能想到的是数据库。当然了数据库是一个方案,那么是否还有其他的解决方案呢?今天我们在讲下Android笔记——Android中数据的存储方式(一) 提到的除了SharedPreferencesFiles(文本文件)以外的其他几种数据储存方式:xml文件SQLite数据Network

1.3  例子

  3.  xml:

    3.1生成xml小案例:  

    下面我们有这样一个小案例:就是短信备份。我们先分析一条短信的结构(如下图)。

  我们看到一条短信包括:短信内容短信发送或接受的时间对方号码类型type(1为接受,2为发送)四种属性(字段)。试着用之前讲过SharedPreferences和Files(文本文件)分析怎么备份?由于SharedPreferences保存的数据只是简单的键值对形式,相对于短信这种结构复杂一些的,他显然是没法去储存的。Files倒是可以做到,定义一个结构格式去保存,但在读写的时候就变得会非常麻烦没有效率。

  •   XML备份原理:目前手机助手短信备份方式虽然多种,但XML格式仍然是比较经典的一种。把短信的全部按照一定的标签格式,写到XML文件中去,再把其保存到本地。从其原理可以看到我首先的就是要生成XML文件。
  •   XML备份短信:

    首先介绍下它保存信息的格式:头文件、根节点(包括开始节点和结束节点)、子节点以及的他的属性等。

  •   布局文件:

    •   

      1. <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
      2. xmlns:tools="http://schemas.android.com/tools"
      3. android:layout_width="match_parent"
      4. android:layout_height="match_parent"
      5. tools:context="${relativePackage}.${activityClass}" >
      6.  
      7. <Button
      8. android:layout_width="wrap_content"
      9. android:layout_height="wrap_content"
      10. android:text="生成XML"/>
      11.  
      12. </RelativeLayout>
  •   java代码:
    •   如何获取系统所保存的短信,为了简介展示,这里就不用内容提供者了,我这里用for循环直接虚拟一个组短信。我们知道手机里的短信少则几条多则上千条,每条短信有四个独立属性,那么我们可以给每条短信封装成一个javabean对象,每个对象有四个常规属性。
    • Sms.java(javabean对象)
      1. package com.bokeyuan.createxml.domain;
      2.  
      3. /**
      4. * 短信内容属性的JavaBean
      5. * @author
      6. *
      7. */
      8. public class Sms {
      9.  
      10. private String address;
      11. private String date;
      12. private String type;
      13. private String body;
      14.  
      15. public String getAddress() {
      16. return address;
      17. }
      18. public void setAddress(String address) {
      19. this.address = address;
      20. }
      21. public String getDate() {
      22. return date;
      23. }
      24. public void setDate(String date) {
      25. this.date = date;
      26. }
      27. public String getType() {
      28. return type;
      29. }
      30. public void setType(String type) {
      31. this.type = type;
      32. }
      33. public String getBody() {
      34. return body;
      35. }
      36. public void setBody(String body) {
      37. this.body = body;
      38. }
      39. public Sms(String address, String date, String type, String body) {
      40. super();
      41. this.address = address;
      42. this.date = date;
      43. this.type = type;
      44. this.body = body;
      45. }
      46.  
      47. @Override
      48. public String toString() {
      49. return "Sms [address=" + address + ", date=" + date + ", type=" + type
      50. + ", body=" + body + "]";
      51. }
      52.  
      53. }
    •  MainActivity.java
      1. package com.bokeyuan.createxml;
      2.  
      3. import java.io.File;
      4. import java.io.FileOutputStream;
      5. import java.io.IOException;
      6. import java.util.ArrayList;
      7. import java.util.List;
      8.  
      9. import com.bokeyuan.createxml.domain.Sms;
      10.  
      11. import android.app.Activity;
      12. import android.os.Bundle;
      13. import android.view.View;
      14.  
      15. public class MainActivity extends Activity {
      16.  
      17. private List<Sms> smslist;
      18.  
      19. @Override
      20. protected void onCreate(Bundle savedInstanceState) {
      21. super.onCreate(savedInstanceState);
      22. setContentView(R.layout.activity_main);
      23.  
      24. smslist = new ArrayList<Sms>();
      25. //假设10条短信
      26. for (int i = 0; i < 10; i++) {
      27. Sms sms = new Sms("110" +i+i, System.currentTimeMillis() + "", "1", "你好,同志" +i);
      28. smslist.add(sms);
      29. }
      30. }
      31.  
      32. public void onClick(View v){
      33. //
      34. StringBuffer sb = new StringBuffer();
      35. //添加属性到sb中
      36. sb.append("<?xml version=\"1.0\" encoding=\"utf-8\"?>");
      37. sb.append("<messages>");
      38. for (Sms sms : smslist) {
      39.        sb.append("<message>");
      40.  
      41. sb.append("<address>");
      42. sb.append(sms.getAddress());
      43. sb.append("</address>");
      44.  
      45. sb.append("<date>");
      46. sb.append(sms.getDate());
      47. sb.append("</date>");
      48.  
      49. sb.append("<type>");
      50. sb.append(sms.getType());
      51. sb.append("</type>");
      52.  
      53. sb.append("<body>");
      54. sb.append(sms.getBody());
      55. sb.append("</body>");
      56.  
      57. sb.append("</message>");
      58. }
      59. sb.append("</messages>");
      60.  
      61. //写入外出储存路径
      62. File file = new File("strorage/sdcard/sms.xml");
      63. try {
      64. FileOutputStream fos = new FileOutputStream(file);
      65. fos.write(sb.toString().getBytes());
      66. fos.close();
      67. } catch (Exception e) {
      68. // TODO Auto-generated catch block
      69. e.printStackTrace();
      70. }
      71. }
      72. }
    • 权限:AndroidManifest.xml中添加android.permission.WRITE_EXTERNAL_STORAGE
  • 问题:实际开放中,当然不会像上面那样拼接字符串生成xml文件,它是很大有弊端的,如下图:
      • 生成xml文件,用浏览器打开,那么就会出现问题了:OPening and ending tag mismatch.
      • 其实谷歌有自己一套自己的生成解析xml的API,使用序列化器XmlSerializer生成xml就是其中的API

  3.2使用序列化器XmlSerializer生成xml

  • 布局文件:同上
  • java代码:    

    • Sms.java(javabean对象):同上  
    • MainActivity.java  
  1. package com.bokeyuan.xmlserilizer;
  2.  
  3. import java.io.File;
  4. import java.io.FileOutputStream;
  5. import java.io.IOException;
  6. import java.util.ArrayList;
  7. import java.util.List;
  8.  
  9. import org.xmlpull.v1.XmlSerializer;
  10.  
  11. import com.bokeyuan.createxml.domain.Sms;
  12.  
  13. import android.os.Bundle;
  14. import android.util.Xml;
  15. import android.view.View;
  16. import android.app.Activity;
  17.  
  18. public class MainActivity extends Activity {
  19.  
  20. List<Sms> smsList;
  21. @Override
  22. protected void onCreate(Bundle savedInstanceState) {
  23. super.onCreate(savedInstanceState);
  24. setContentView(R.layout.activity_main);
  25.  
  26. smsList = new ArrayList<Sms>();
  27.  
  28. //虚构10条短信
  29. for (int i = 0; i < 10; i++) {
  30. Sms sms = new Sms("138"+i+i, System.currentTimeMillis() + "", "1", "哈哈"+i);
  31. smsList.add(sms);
  32. }
  33. }
  34.  
  35. public void click(View v){
  36. //使用xml序列化器生成xml文件
  37. //1.拿到序列化器对象
  38. XmlSerializer xs = Xml.newSerializer();
  39.  
  40. File file = new File("sdcard/sms2.xml");
  41. try {
  42. //2.对序列化器进行初始化
  43. FileOutputStream fos = new FileOutputStream(file);
  44. //OutputStream :指定文件的保存路径
  45. //encoding:指定生成的xml文件的编码
  46. xs.setOutput(fos, "utf-8");
  47.  
  48. //3.开始生成文件
  49. //生成头结点
  50. xs.startDocument("utf-8", true);
  51. //生成开始标签
  52. xs.startTag(null, "messages");
  53.  
  54. for (Sms sms : smsList) {
  55. xs.startTag(null, "message");
  56.  
  57. xs.startTag(null, "address");
  58. //生成文本节点
  59. xs.text(sms.getAddress());
  60. xs.endTag(null, "address");
  61.  
  62. xs.startTag(null, "date");
  63. //生成文本节点
  64. xs.text(sms.getDate());
  65. xs.endTag(null, "date");
  66.  
  67. xs.startTag(null, "type");
  68. //生成文本节点
  69. xs.text(sms.getType());
  70. xs.endTag(null, "type");
  71.  
  72. xs.startTag(null, "body");
  73. //生成文本节点
  74. xs.text(sms.getBody() + "<body>");
  75. xs.endTag(null, "body");
  76.  
  77. xs.endTag(null, "message");
  78. }
  79.  
  80. //生成结束标签
  81. xs.endTag(null, "messages");
  82.  
  83. //告诉序列化器,生成完毕
  84. xs.endDocument();
  85. } catch (Exception e) {
  86. // TODO Auto-generated catch block
  87. e.printStackTrace();
  88. }
  89. }
  90. }
  • 权限:AndroidManifest.xml中添加android.permission.WRITE_EXTERNAL_STORAGE
  • 解决问题:
    •   同样,添加一个字符串"<body>",导出xml文件。
    •   

     用浏览器打开,发现没有报错,随意添加的字符串"<body>",做为普通文本,而不是标签显示出来了。如下图:

      

    这是因为,XmlSerializer序列化把字符串"<body>"  做一个字符的转义,我么把生成的xml文件用文本文件打开,可以看到:

    

  

  3.3 XML解析:

  •   这里我们是用Pull解析XML。Pull解析器是一个开源项目,既可以用在Android应用,亦可用在JavaEE上。如果需要在JavaEE应用中使用Pull解析,需要添加Pull解析器的JAR包。但是Android开发平台已经内置了Pull解析器,并且Android系统本身也使用Pull解析器解析各种XML文档,因此Android推荐开发者使用Pull解析器来解析XML文档。
  •   除了Pull解析之外,Java开发者还可使用DOM或SAX对XML进行解析。一般的Java应用会使用JAXP API来解析XML。在实际开发中,使用JDOM或dom4j进行解析可能更加简单。
  •   应用场景:一是解析XML格式的备份数据之类的,二是客户端向服务器请求数据,当数据内容比较多、结构比较复杂的时候,服务器按照一定的格式会把数据进行封装,再把封装之后的数据传送给客户端。服务器封装数据的格式很多,其中的Android中常解析的格式就是XMLJSON
  •  下面案例:查询天气的功能,自动显示在界面上。

  XML资源的内容如下:这里放在项目src下

      

  • 布局文件:

    1. <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    2. xmlns:tools="http://schemas.android.com/tools"
    3. android:layout_width="match_parent"
    4. android:layout_height="match_parent"
    5. android:paddingBottom="@dimen/activity_vertical_margin"
    6. android:paddingLeft="@dimen/activity_horizontal_margin"
    7. android:paddingRight="@dimen/activity_horizontal_margin"
    8. android:paddingTop="@dimen/activity_vertical_margin"
    9. tools:context=".MainActivity" >
    10.  
    11. <Button
    12. android:layout_width="wrap_content"
    13. android:layout_height="wrap_content"
    14. android:text="解析xml"
    15. android:onClick="click"
    16. />
    17.  
    18. </RelativeLayout>
  • java代码:    
    • City.java(javabean对象):

      1. package com.bokeyuan.pullparse.domain;
      2.  
      3. public class City {
      4.  
      5. private String name;
      6. private String temp;
      7. private String pm25;
      8. public String getName() {
      9. return name;
      10. }
      11. public void setName(String name) {
      12. this.name = name;
      13. }
      14. public String getTemp() {
      15. return temp;
      16. }
      17. public void setTemp(String temp) {
      18. this.temp = temp;
      19. }
      20. public String getPm25() {
      21. return pm25;
      22. }
      23. public void setPm25(String pm25) {
      24. this.pm25 = pm25;
      25. }
      26. public City(String name, String temp, String pm25) {
      27. super();
      28. this.name = name;
      29. this.temp = temp;
      30. this.pm25 = pm25;
      31. }
      32. public City() {
      33. super();
      34. }
      35. @Override
      36. public String toString() {
      37. return "City [name=" + name + ", temp=" + temp + ", pm25=" + pm25 + "]";
      38. }
      39. }

        

    • MainActivity.java 
      1. package com.yuanboyuan.pullparse;
      2.  
      3. import java.io.InputStream;
      4. import java.util.ArrayList;
      5. import java.util.List;
      6.  
      7. import org.xmlpull.v1.XmlPullParser;
      8. import org.xmlpull.v1.XmlPullParserException;
      9.  
      10. import com.yuanboyuan.pullparse.domain.City;
      11.  
      12. import android.os.Bundle;
      13. import android.app.Activity;
      14. import android.util.Xml;
      15. import android.view.Menu;
      16. import android.view.View;
      17.  
      18. public class MainActivity extends Activity {
      19.  
      20. List<City> cityList;
      21. @Override
      22. protected void onCreate(Bundle savedInstanceState) {
      23. super.onCreate(savedInstanceState);
      24. setContentView(R.layout.activity_main);
      25. }
      26. public void click(View v){
      27. //解析xml文件
      28. //1. 拿到资源文件
      29. InputStream is = getClassLoader().getResourceAsStream("weather.xml");
      30.  
      31. //2. 拿到解析器对象
      32. XmlPullParser xp = Xml.newPullParser();
      33. try {
      34. //3. 初始化xp对象
      35. xp.setInput(is, "gbk");
      36.  
      37. //4.开始解析
      38. //获取当前节点的事件类型
      39. int type = xp.getEventType();
      40. City city = null;
      41. while(type != XmlPullParser.END_DOCUMENT){
      42. //判断当前解析到哪一个节点,从而确定你要做什么操作
      43. switch (type) {
      44. case XmlPullParser.START_TAG:
      45. // 获取当前节点的名字
      46. if("weather".equals(xp.getName())){
      47. cityList = new ArrayList<City>();
      48. }
      49. else if("city".equals(xp.getName())){
      50. city = new City();
      51. }
      52. else if("name".equals(xp.getName())){
      53. // 获取当前节点的下一个节点的文本,把指针移动到当前节点的结束节点
      54. String name = xp.nextText();
      55. city.setName(name);
      56. }
      57. else if("temp".equals(xp.getName())){
      58. // 获取当前节点的下一个节点的文本,把指针移动到当前节点的结束节点
      59. String temp = xp.nextText();
      60. city.setTemp(temp);
      61. }
      62. else if("pm25".equals(xp.getName())){
      63. // 获取当前节点的下一个节点的文本,把指针移动到当前节点的结束节点
      64. String pm25 = xp.nextText();
      65. city.setPm25(pm25);
      66. }
      67. break;
      68. case XmlPullParser.END_TAG:
      69. if("city".equals(xp.getName())){
      70. cityList.add(city);
      71. }
      72. break;
      73. }
      74. //把指针移动到下一个节点
      75. type = xp.next();
      76. }
      77. for (City c : cityList) {
      78. System.out.println(c.toString());
      79. }
      80. } catch (Exception e) {
      81. // TODO Auto-generated catch block
      82. e.printStackTrace();
      83. }
      84. }
      85. }

  

  

参考资料:

Android应用开发基础之数据存储和界面展现(三)

《疯狂Android讲义》(第2版)

Android笔记——Android中数据的存储方式(二)的更多相关文章

  1. Android笔记——Android中数据的存储方式(一)

    Android中数据的存储方式 对于开发平台来讲,如果对数据的存储有良好的支持,那么对应用程序的开发将会有很大的促进作用. 总体的来讲,数据存储方式有三种:一个是文件,一个是数据库,另一个则是网络.其 ...

  2. Matlab中数据的存储方式

    简介 MATLAB提供了丰富的算法以及一个易于操作的语言,给算法研发工作者提供了很多便利.然而MATLAB在执行某些任务的时候,执行效率偏低,测试较大任务量时可能会引起较长时间的等待.未解决这个问题, ...

  3. Android笔记——Android中数据的存储方式(三)

    Android系统集成了一个轻量级的数据库:SQLite,所以Android对数据库的支持很好,每个应用都可以方便的使用它.SQLite作为一个嵌入式的数据库引擎,专门适用于资源有限的设备上适量数据存 ...

  4. Android的数据的存储方式

    数据的存储方式,总的来说分为三种: ① 文件存储: * SharedPreferences存储 * SD卡存储 ---- Environment * 数据库存储 ---- SQLite .MySQL. ...

  5. C语言中float,double类型,在内存中的结构(存储方式)

    C语言中float,double类型,在内存中的结构(存储方式)从存储结构和算法上来讲,double和float是一样的,不一样的地方仅仅是float是32位的,double是64位的,所以doubl ...

  6. Android笔记--View绘制流程源码分析(二)

    Android笔记--View绘制流程源码分析二 通过上一篇View绘制流程源码分析一可以知晓整个绘制流程之前,在activity启动过程中: Window的建立(activit.attach生成), ...

  7. 讨论两种Redis中Token的存储方式

    摘要:本文讨论一个问题:存储token时,token与对应用户id谁来作为key? 问题起源问题起源于要给公司的后台管理系统添加权限管理,选用的是开源框架shiro,而原本系统上是采用token做了登 ...

  8. 原码,补码,反码的概念及Java中使用那种存储方式

    原码,补码,反码的概念及Java中使用那种存储方式: 原码:原码表示法是机器数的一种简单的表示法.其符号位用0表示正号,用:表示负号,数值一般用二进制形式表示 补码:机器数的补码可由原码得到.如果机器 ...

  9. 将Excel中数据导入数据库(二)

    在上篇文章中介绍到将Excel中数据导入到数据库中,但上篇文章例子只出现了nvachar类型,且数据量很小.今天碰到将Excel中数据导入数据库中的Excel有6419行,其中每行均有48个字段,有i ...

随机推荐

  1. Oracle 行转列总结 Case When,Decode,PIVOT 三种方式 - 转

    最近又碰到行专列问题了,当时不假思索用的是子查询,做完后我询问面试管行专列标正的写法应该如何写,他告诉我说应该用"Decode",索性我就总结一下,一共三种方式 --======= ...

  2. [转]两种Sigma-Delta ADC SNR仿真方法

    假设现有一组Sigma-Delta ADC输出序列,下面将介绍两种计算出相应SNR的方法.其中由cadence导出数据的CIW窗口命令为:ocPrint(?output "输出目录/输出文件 ...

  3. 基于SWFUpload的angular上传组件

    回顾 由于工作内容比较多,特别是架构方面,需要耗费很多的时间调整.重构,因此很久没有写文章了. 话就不多说了,直接进入主题. 实现 首先分析一下SWFUpload初始化的时候,需要传入当前触发上传的元 ...

  4. fusioncharts图例(legend)属性

    图例用来在多系列图和混合图中将图形和对应的系列名称联系起来.      从v3.2开始,每个系列的名称前面会展示对应的icon图标,这些图标具有交互作用,用户可以通过点击这些图标来显示或者隐藏对应的数 ...

  5. 用 .NET Reflector 8 查看 System.Windows.Controls 命名空间下的类

    为了学习自定义控件,就想看看WPF基本元素的代码.使用到工具.NET Reflector. System.Windows.Controls 命名空间在PresentationFramework.dll ...

  6. PHP __DIR__, __FILE__, __FUNCTION__, __CLASS__, __METHOD__, __LINE__, __NAMESPACE__

    PHP has large number of predefined constants. This HOWTO will present the seven most important, most ...

  7. webpack多页面开发与懒加载hash解决方案

    之前讨论了webpack的hash与chunkhash的区别以及各自的应用场景,如果是常规单页面应用的话,上篇文章提供的方案是没有问题的.但是前端项目复杂多变,应对复杂多页面项目时,我们不得不继续踩w ...

  8. MySQL忘记root密码的找回方法

    (1)登录到数据库所在服务器,手工kill掉MySQL进程: kill ' cat /mysql-data-directory/hostname.pid'     其中,/mysql-data-dir ...

  9. Android学习笔记之横向二级菜单实现

    PS:元旦来一发. 学习内容: 1.Android二级横向菜单的实现过程.效果如上图...   这种横向的二级菜单在很多的app都有所应用.效果看起来还是非常的美观的.也算是项目需要,自己也就学了一下 ...

  10. node生成自定义命令(yargs/commander)

    第一部分可以生成一个自定义命令,例如常见的”express”,yargs和commander则可以在生成的自定义命令上做扩展,yargs将命令扩展成类似express --l xx的形式;而comma ...