Android笔记——Android中数据的存储方式(二)

时间:2021-10-02 22:22:48

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

1.3  例子

  3.  xml:

    3.1生成xml小案例:  

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

Android笔记——Android中数据的存储方式(二)

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

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

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

Android笔记——Android中数据的存储方式(二)

  •   布局文件:
    •   
      <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
      xmlns:tools="http://schemas.android.com/tools"
      android:layout_width="match_parent"
      android:layout_height="match_parent"
      tools:context="${relativePackage}.${activityClass}" > <Button
      android:layout_width="wrap_content"
      android:layout_height="wrap_content"
      android:text="生成XML"/> </RelativeLayout>
  •   java代码:
    •   如何获取系统所保存的短信,为了简介展示,这里就不用内容提供者了,我这里用for循环直接虚拟一个组短信。我们知道手机里的短信少则几条多则上千条,每条短信有四个独立属性,那么我们可以给每条短信封装成一个javabean对象,每个对象有四个常规属性。
    • Sms.java(javabean对象)
      package com.bokeyuan.createxml.domain;
      
      /**
      * 短信内容属性的JavaBean
      * @author
      *
      */
      public class Sms { private String address;
      private String date;
      private String type;
      private String body; public String getAddress() {
      return address;
      }
      public void setAddress(String address) {
      this.address = address;
      }
      public String getDate() {
      return date;
      }
      public void setDate(String date) {
      this.date = date;
      }
      public String getType() {
      return type;
      }
      public void setType(String type) {
      this.type = type;
      }
      public String getBody() {
      return body;
      }
      public void setBody(String body) {
      this.body = body;
      }
      public Sms(String address, String date, String type, String body) {
      super();
      this.address = address;
      this.date = date;
      this.type = type;
      this.body = body;
      } @Override
      public String toString() {
      return "Sms [address=" + address + ", date=" + date + ", type=" + type
      + ", body=" + body + "]";
      } }
    •  MainActivity.java
      package com.bokeyuan.createxml;
      
      import java.io.File;
      import java.io.FileOutputStream;
      import java.io.IOException;
      import java.util.ArrayList;
      import java.util.List; import com.bokeyuan.createxml.domain.Sms; import android.app.Activity;
      import android.os.Bundle;
      import android.view.View; public class MainActivity extends Activity { private List<Sms> smslist; @Override
      protected void onCreate(Bundle savedInstanceState) {
      super.onCreate(savedInstanceState);
      setContentView(R.layout.activity_main); smslist = new ArrayList<Sms>();
      //假设10条短信
      for (int i = 0; i < 10; i++) {
      Sms sms = new Sms("110" +i+i, System.currentTimeMillis() + "", "1", "你好,同志" +i);
      smslist.add(sms);
      }
      } public void onClick(View v){
      //
      StringBuffer sb = new StringBuffer();
      //添加属性到sb中
      sb.append("<?xml version=\"1.0\" encoding=\"utf-8\"?>");
      sb.append("<messages>");
      for (Sms sms : smslist) {
             sb.append("<message>"); sb.append("<address>");
      sb.append(sms.getAddress());
      sb.append("</address>"); sb.append("<date>");
      sb.append(sms.getDate());
      sb.append("</date>"); sb.append("<type>");
      sb.append(sms.getType());
      sb.append("</type>"); sb.append("<body>");
      sb.append(sms.getBody());
      sb.append("</body>"); sb.append("</message>");
      }
      sb.append("</messages>"); //写入外出储存路径
      File file = new File("strorage/sdcard/sms.xml");
      try {
      FileOutputStream fos = new FileOutputStream(file);
      fos.write(sb.toString().getBytes());
      fos.close();
      } catch (Exception e) {
      // TODO Auto-generated catch block
      e.printStackTrace();
      }
      }
      }
    • 权限:AndroidManifest.xml中添加android.permission.WRITE_EXTERNAL_STORAGE
  • 问题:实际开放中,当然不会像上面那样拼接字符串生成xml文件,它是很大有弊端的,如下图:
      • Android笔记——Android中数据的存储方式(二)
      • 生成xml文件,用浏览器打开,那么就会出现问题了:OPening and ending tag mismatch.
      • Android笔记——Android中数据的存储方式(二)
      • 其实谷歌有自己一套自己的生成解析xml的API,使用序列化器XmlSerializer生成xml就是其中的API

  3.2使用序列化器XmlSerializer生成xml

  • 布局文件:同上
  • java代码:    
    • Sms.java(javabean对象):同上  
    • MainActivity.java  
package com.bokeyuan.xmlserilizer;

import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.util.ArrayList;
import java.util.List; import org.xmlpull.v1.XmlSerializer; import com.bokeyuan.createxml.domain.Sms; import android.os.Bundle;
import android.util.Xml;
import android.view.View;
import android.app.Activity; public class MainActivity extends Activity { List<Sms> smsList;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main); smsList = new ArrayList<Sms>(); //虚构10条短信
for (int i = 0; i < 10; i++) {
Sms sms = new Sms("138"+i+i, System.currentTimeMillis() + "", "1", "哈哈"+i);
smsList.add(sms);
}
} public void click(View v){
//使用xml序列化器生成xml文件
//1.拿到序列化器对象
XmlSerializer xs = Xml.newSerializer(); File file = new File("sdcard/sms2.xml");
try {
//2.对序列化器进行初始化
FileOutputStream fos = new FileOutputStream(file);
//OutputStream :指定文件的保存路径
//encoding:指定生成的xml文件的编码
xs.setOutput(fos, "utf-8"); //3.开始生成文件
//生成头结点
xs.startDocument("utf-8", true);
//生成开始标签
xs.startTag(null, "messages"); for (Sms sms : smsList) {
xs.startTag(null, "message"); xs.startTag(null, "address");
//生成文本节点
xs.text(sms.getAddress());
xs.endTag(null, "address"); xs.startTag(null, "date");
//生成文本节点
xs.text(sms.getDate());
xs.endTag(null, "date"); xs.startTag(null, "type");
//生成文本节点
xs.text(sms.getType());
xs.endTag(null, "type"); xs.startTag(null, "body");
//生成文本节点
xs.text(sms.getBody() + "<body>");
xs.endTag(null, "body"); xs.endTag(null, "message");
} //生成结束标签
xs.endTag(null, "messages"); //告诉序列化器,生成完毕
xs.endDocument();
} catch (Exception e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
  • 权限:AndroidManifest.xml中添加android.permission.WRITE_EXTERNAL_STORAGE
  • 解决问题:
    •   同样,添加一个字符串"<body>",导出xml文件。
    •   Android笔记——Android中数据的存储方式(二)

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

      Android笔记——Android中数据的存储方式(二)

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

    Android笔记——Android中数据的存储方式(二)

  

  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下

      Android笔记——Android中数据的存储方式(二)

  • 布局文件:
    <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:paddingBottom="@dimen/activity_vertical_margin"
    android:paddingLeft="@dimen/activity_horizontal_margin"
    android:paddingRight="@dimen/activity_horizontal_margin"
    android:paddingTop="@dimen/activity_vertical_margin"
    tools:context=".MainActivity" > <Button
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:text="解析xml"
    android:onClick="click"
    /> </RelativeLayout>
  • java代码:    
    • City.java(javabean对象):
      package com.bokeyuan.pullparse.domain;
      
      public class City {
      
          private String name;
      private String temp;
      private String pm25;
      public String getName() {
      return name;
      }
      public void setName(String name) {
      this.name = name;
      }
      public String getTemp() {
      return temp;
      }
      public void setTemp(String temp) {
      this.temp = temp;
      }
      public String getPm25() {
      return pm25;
      }
      public void setPm25(String pm25) {
      this.pm25 = pm25;
      }
      public City(String name, String temp, String pm25) {
      super();
      this.name = name;
      this.temp = temp;
      this.pm25 = pm25;
      }
      public City() {
      super();
      }
      @Override
      public String toString() {
      return "City [name=" + name + ", temp=" + temp + ", pm25=" + pm25 + "]";
      }
      }

        

    • MainActivity.java 
      package com.yuanboyuan.pullparse;
      
      import java.io.InputStream;
      import java.util.ArrayList;
      import java.util.List; import org.xmlpull.v1.XmlPullParser;
      import org.xmlpull.v1.XmlPullParserException; import com.yuanboyuan.pullparse.domain.City; import android.os.Bundle;
      import android.app.Activity;
      import android.util.Xml;
      import android.view.Menu;
      import android.view.View; public class MainActivity extends Activity { List<City> cityList;
      @Override
      protected void onCreate(Bundle savedInstanceState) {
      super.onCreate(savedInstanceState);
      setContentView(R.layout.activity_main);
      }
      public void click(View v){
      //解析xml文件
      //1. 拿到资源文件
      InputStream is = getClassLoader().getResourceAsStream("weather.xml"); //2. 拿到解析器对象
      XmlPullParser xp = Xml.newPullParser();
      try {
      //3. 初始化xp对象
      xp.setInput(is, "gbk"); //4.开始解析
      //获取当前节点的事件类型
      int type = xp.getEventType();
      City city = null;
      while(type != XmlPullParser.END_DOCUMENT){
      //判断当前解析到哪一个节点,从而确定你要做什么操作
      switch (type) {
      case XmlPullParser.START_TAG:
      // 获取当前节点的名字
      if("weather".equals(xp.getName())){
      cityList = new ArrayList<City>();
      }
      else if("city".equals(xp.getName())){
      city = new City();
      }
      else if("name".equals(xp.getName())){
      // 获取当前节点的下一个节点的文本,把指针移动到当前节点的结束节点
      String name = xp.nextText();
      city.setName(name);
      }
      else if("temp".equals(xp.getName())){
      // 获取当前节点的下一个节点的文本,把指针移动到当前节点的结束节点
      String temp = xp.nextText();
      city.setTemp(temp);
      }
      else if("pm25".equals(xp.getName())){
      // 获取当前节点的下一个节点的文本,把指针移动到当前节点的结束节点
      String pm25 = xp.nextText();
      city.setPm25(pm25);
      }
      break;
      case XmlPullParser.END_TAG:
      if("city".equals(xp.getName())){
      cityList.add(city);
      }
      break;
      }
      //把指针移动到下一个节点
      type = xp.next();
      }
      for (City c : cityList) {
      System.out.println(c.toString());
      }
      } catch (Exception e) {
      // TODO Auto-generated catch block
      e.printStackTrace();
      }
      }
      }

  

  

参考资料:

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&comma;double类型&comma;在内存中的结构&lpar;存储方式&rpar;

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

  6. Android笔记--View绘制流程源码分析&lpar;二&rpar;

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

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

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

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

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

  9. Android开发-之数据的存储方式一

    在Android中,数据的存储分为两种方式: 1.直接以文件的形式存储在目录中 2.以json格式存储在数据库中 将数据以文件的存储又分为两种方式: 1.生成.txt文件 2.生成xml文件 那么今天 ...

随机推荐

  1. Java关键字final、static

    一.final根据程序上下文环境,Java关键字final有“这是无法改变的”或者“终态的”含义,它可以修饰非抽象类.非抽象类成员方法和变量.你可能出于两种理解而需要阻止改变:设计或效率. final ...

  2. twitter storm源码走读之1 -- nimbus启动场景分析

    欢迎转载,转载时请注明作者徽沪一郎及出处,谢谢. 本文详细介绍了twitter storm中的nimbus节点的启动场景,分析nimbus是如何一步步实现定义于storm.thrift中的servic ...

  3. android 原生dialog对话框

    http://www.cnblogs.com/xiaoluo501395377/p/3419398.html

  4. &period;net类库中和数据库相关的

    System.Data.SqlTypes SqlDbType 枚举类型 Specifies SQL Server-specific data type of a field, property, fo ...

  5. Unity3D--学习太空射击游戏制作(一)

    近期买了本书在学习一些Unity3D的东西,在了解了Unity3D工具的基本面板后开始学习一个太空射击游戏的开发过程. 首先下载一个关于本游戏的资源文件,(百度云下载地址:http://pan.bai ...

  6. javax&period;servlet-api 和 servlet-api 区别

    作者:Night Silent链接:https://www.zhihu.com/question/35812310/answer/64578510来源:知乎著作权归作者所有.商业转载请联系作者获得授权 ...

  7. 使用Stickers拓展集成iMessage简单功能

    添加一个target,选择Stickers拓展: 然后就会出现iMessage的文件夹:添加你需要的iMessage图片,这里图片遵循下面的要求: Small: 100 x 100 pt @3x sc ...

  8. 解决Eclipse 项目报错:Unbound classpath container&colon; &OpenCurlyQuote;JRE System Library &lbrack;JavaSE-1&period;7&rsqb;

    MyEclipse出现下面两条报错: The project cannot be built until build path errors are resolved HelloWord Unknow ...

  9. bzoj 1588 平衡树 splay

    1588: [HNOI2002]营业额统计 Time Limit: 5 Sec  Memory Limit: 162 MBSubmit: 15446  Solved: 6076[Submit][Sta ...

  10. code First 三 Fluent API

    Entity Framework Fluent API用于配置域类以覆盖约定. 在实体框架6中,DbModelBuilder类充当Fluent API,我们可以使用它来配置许多不同的东西.它提供了比数 ...