在网络上传输数据时最常用的方法有两种:XML和JSON,下面就对这两种类型的数据解析进行讲解。

一、XML数据解析

在Android中,常见的XML解析器分别为SAX解析器、DOM解析器和PULL解析器。

(1)SAX解析

SAX(Simple API for XML)解析器是一种基于事件的解析器,它的核心是事件处理模式,主要是围绕着事件源以及事件处理器来工作的。当事件源产生事件后,调用事件处理器相应的处理方法,一个事件就可以得到处理。在事件源调用事件处理器中特定方法的时候,还要传递给事件处理器相应事件的状态信息,这样事件处理器才能够根据提供的事件信息来决定自己的行为。

SAX解析器的优点是解析速度快,占用内存少。非常适合在Android移动设备中使用。

(2)DOM解析

DOM是基于树形结构的的节点或信息片段的集合,允许开发人员使用DOM API遍历XML树、检索所需数据。分析该结构通常需要加载整个文档和构造树形结构,然后才可以检索和更新节点信息。

由于DOM在内存中以树形结构存放,因此检索和更新效率会更高。但是对于特别大的文档,解析和加载整个文档将会很耗资源。

(3)PULL解析

PULL解析器的运行方式和SAX类似,都是基于事件的模式。不同的是,在PULL解析过程中,我们需要自己获取产生的事件然后做相应的操作,而不像SAX那样由处理器触发一种事件的方法,执行我们的代码。PULL解析器小巧轻便,解析速度快,简单易用,非常适合在Android移动设备中使用,Android系统内部在解析各种XML时也是用PULL解析器。

代码实现:

SAX解析实现:

public class SAXParserTool {
private static List<Book> result_books; public static List<Book> parserXMLWithSAX(InputStream is) throws ParserConfigurationException, SAXException, IOException{
SAXParserFactory parserFactory=SAXParserFactory.newInstance();
SAXParser parser=parserFactory.newSAXParser();
MyHandler handler=new MyHandler();
parser.parse(is, handler);
return result_books;
} /**
* 继承DefaultHandler类进行XML处理
* 重写5个方法startDocument(),endDocument(),startElement,endElement,characters
*/
static class MyHandler extends DefaultHandler{
private List<Book> books;
private Book book;
private StringBuilder builder; @Override
public void startDocument() throws SAXException { books = new ArrayList<Book>();
builder = new StringBuilder();
} @Override
public void startElement(String uri, String localName, String qName,
Attributes attributes) throws SAXException {
if(localName.equals("book")){
book=new Book();
}
builder.setLength(0);
} @Override
public void characters(char[] ch, int start, int length)
throws SAXException {
builder.append(ch,start,length);
} @Override
public void endElement(String uri, String localName, String qName)
throws SAXException {
if(localName.equals("id")){
book.setId(builder.toString());
}
if(localName.equals("name")){
book.setName(builder.toString());
}
if(localName.equals("price")){
book.setPrice(builder.toString());
}
if(localName.equals("book")){
books.add(book);
}
} @Override
public void endDocument() throws SAXException {
// TODO Auto-generated method stub
super.endDocument();
result_books=books;
} } }

PULL解析实现:

public class PULLParserTool {
private static List<Book> books;
private static Book book; public static List<Book> parserXMLWithPull(InputStream in) throws XmlPullParserException, IOException{
XmlPullParser parser=Xml.newPullParser();
parser.setInput(in, "UTF-8"); /*
*PULL解析跟SAX解析原理差不多,
*但是SAX解析有定义好的事件触发机制,
*而PUll解析需要自己处理事件触发
*/
int evenType=parser.getEventType();
while(evenType!=XmlPullParser.END_DOCUMENT){
switch (evenType) {
case XmlPullParser.START_DOCUMENT:
books=new ArrayList<Book>();
break; case XmlPullParser.START_TAG:
if(parser.getName().equals("book")){
book=new Book();
}
if(parser.getName().equals("id")){
//要使用nextText()获取该标签下内容
book.setId(parser.nextText());
}
if(parser.getName().equals("name")){
book.setName(parser.nextText());
}
if(parser.getName().equals("price")){
book.setPrice(parser.nextText());
}
break; case XmlPullParser.END_TAG:
if(parser.getName().equals("book")){
books.add(book);
book=null;
}
break;
}
evenType=parser.next();
} return books; }
}

DOM解析实现:

public class DOMParserTool {

    public static List<Book> parserXMLWithDOM(InputStream is) throws ParserConfigurationException, SAXException, IOException{
List<Book> books = new ArrayList<Book>();
DocumentBuilderFactory factory=DocumentBuilderFactory.newInstance();
DocumentBuilder builder=factory.newDocumentBuilder(); /*
* DOM解析会将整个XML文件先读取到内存中以树形结构存放
* 然后遍历该XML树来检索出所需数据
*/
Document document=builder.parse(is);
Element rootElement=document.getDocumentElement();
NodeList items=rootElement.getElementsByTagName("book");
for (int i = 0; i < items.getLength(); i++) {
Book book = new Book();
Node item = items.item(i);
NodeList properties = item.getChildNodes();
for (int j = 0; j < properties.getLength(); j++) {
Node property = properties.item(j);
String nodeName = property.getNodeName();
if (nodeName.equals("id")) {
book.setId(property.getFirstChild().getNodeValue());
} else if (nodeName.equals("name")) {
book.setName(property.getFirstChild().getNodeValue());
} else if (nodeName.equals("price")) {
book.setPrice(property.getFirstChild().getNodeValue());
}
}
books.add(book);
}
return books;
} }

二、Json解析

JSON是JavaScript Object Notation的缩写,可见JSON来源于JavaScript。JSON数据是一系列键值对的集合。

JSON和XML的比较

  • JSON比XML体积小
  • JSON和JavaScript交互更加方便
  • JSON对数据的描述性没有XML好
  • JSON的速度要远远大于XML
  • JSON的解析要比XML的解析要方便

JSON已经被大多数开发人员所接受,在网络数据的传输当中应用非常广泛。

1、使用JSONObject

(1)使用JSONObject创建json字符串

           JSONObject person = new JSONObject();
JSONArray phone = new JSONArray();
phone.put("12345678");
phone.put("87654321");
person.put("phone", phone);
// 第2个value一个字符串
person.put("name", "tianjiefeng");
// 第3个value一个字符串数值
person.put("age", 100);
// 第4个value一个JSONObject对象
JSONObject address = new JSONObject();
address.put("country", "china");
address.put("province", "jiangsu");
person.put("address", address);
// 第5个value一个布尔值
person.put("married", false);
//获取JSON字符串
String JSONStr=person.toString();
JSON字符串:
{
"phone" : ["", ""], // 数组
"name" : "tianjiefeng", // 字符串
"age" : , // 数值
"address" : { "country" : "china", "province" : "jiangsu" }, // JSON对象
"married" : false // 布尔值
}

(2)使用JSONObject解析JSON字符串为对象

private Map<String, Object> parseJSONString(String JSONString) {
Map<String, Object> resultMap = new HashMap<String, Object>();
try {
// 直接把JSON字符串转化为一个JSONObject对象
JSONObject person = new JSONObject(JSONString);
// 第1个键值对
resultMap.put("phone", person.getJSONArray("phone").toString());
// 第2个键值对
resultMap.put("name", person.getString("name"));
// 第3个键值对
resultMap.put("age", person.getInt("age"));
// 第4个键值对
resultMap.put("address", person.getJSONObject("address").toString());
// 第5个键值对
resultMap.put("married", person.getBoolean("married"));
} catch (JSONException e) {
e.printStackTrace();
}
return resultMap;
}

 2、使用GSON

相对于较为传统的Json解析来说,google共享的开源Gson在解析速度和所使用的内存在有着明显的优势.

在GSON的API中,提供了两个重要的方法:toJson()和fromJson()方法。其中,toJson()方法用来实现将Java对象转换为相应的JSON数据,fromJson()方法则用来实现将JSON数据转换为相应的Java对象。

toJson()方法

  toJson()方法用于将Java对象转换为相应的JSON数据,主要有以下几种形式:

  (1)String toJson(JsonElement jsonElement);

  (2)String toJson(Object src);

  (3)String toJson(Object src, Type typeOfSrc);

  其中,方法(1)用于将JsonElement对象(可以是JsonObject、JsonArray等)转换成JSON数据;方法(2)用于将指定的Object对象序列化成相应的JSON数据;方法(3)用于将指定的Object对象(可以包括泛型类型)序列化成相应的JSON数据。

fromJson()方法

  fromJson()方法用于将JSON数据转换为相应的Java对象,主要有以下几种形式:

  (1)<T> T fromJson(JsonElement json, Class<T> classOfT);

  (2)<T> T fromJson(JsonElement json, Type typeOfT);

  (3)<T> T fromJson(JsonReader reader, Type typeOfT);

  (4)<T> T fromJson(Reader reader, Class<T> classOfT);

  (5)<T> T fromJson(Reader reader, Type typeOfT);

  (6)<T> T fromJson(String json, Class<T> classOfT);

  (7)<T> T fromJson(String json, Type typeOfT);

  以上的方法用于将不同形式的JSON数据解析成Java对象。

android基础(五)网络数据解析方法的更多相关文章

  1. Android基础(五) Service全解析----看不见的Activity

    一.服务的介绍: 作为Android四大组件之中的一个,Service(服务)也常常运用于我们的日常使用中,它与Activity的差别在于:Service一直在后台执行.没实用户界面.所以绝不会到前台 ...

  2. Android中级之网络数据解析一之xml解析

    本文来自http://blog.csdn.net/liuxian13183/ ,引用必须注明出处! --Comic Sans MS Xml解析具有跨平台性,语言无关性,易操作性,因此广受开发者的欢迎. ...

  3. (转载)Android之三种网络请求解析数据(最佳案例)

    [置顶] Android之三种网络请求解析数据(最佳案例) 2016-07-25 18:02 4725人阅读 评论(0) 收藏 举报  分类: Gson.Gson解析(1)  版权声明:本文为博主原创 ...

  4. wemall app商城源码Android 获取XML网络数据并绑定到ListView

    wemall-mobile是基于WeMall的android app商城,只需要在原商城目录下上传接口文件即可完成服务端的配置,客户端可定制修改.本文分享Android 获取XML网络数据并绑定到Li ...

  5. 浅议iOS网络数据解析

    /*------------------------------------ 数据解析: 1.JSON数据 --------------------------------*/ 重点:1.什么是JSO ...

  6. JSON三种数据解析方法(转)

    原 JSON三种数据解析方法 2018年01月15日 13:05:01 zhoujiang2012 阅读数:7896    版权声明:本文为博主原创文章,未经博主允许不得转载. https://blo ...

  7. Android之三种网络请求解析数据(最佳案例)

    AsyncTask解析数据 AsyncTask主要用来更新UI线程,比较耗时的操作可以在AsyncTask中使用. AsyncTask是个抽象类,使用时需要继承这个类,然后调用execute()方法. ...

  8. Python黑客编程基础3网络数据监听和过滤

    网络数据监听和过滤 课程的实验环境如下: •      操作系统:kali Linux 2.0 •      编程工具:Wing IDE •      Python版本:2.7.9 •      涉及 ...

  9. Android中获取网络数据时的分页加载

    //此实在Fragment中实现的,黄色部分为自动加载,红色部分是需要注意的和手动加载,    蓝色部分是睡眠时间,自我感觉不用写  ,还有就是手动加载时,不知道为什么进去后显示的就是最后一行,求大神 ...

随机推荐

  1. 在iframe父界面获取iframe里面的标签

    上一篇里边介绍了在里边利用iframe引入另一个html导航文件,有兴趣的朋友可以看一看 http://www.cnblogs.com/simba-lkj/p/6031662.html 目前遇到一些问 ...

  2. hihocoder-平衡树·SBT

    http://hihocoder.com/problemset/problem/1337 #1337 : 平衡树·SBT 时间限制:10000ms 单点时限:1000ms 内存限制:256MB 描述 ...

  3. 关于nginx反向代理后获取不到客户端的真实ip地址问题

    前段时间在我的网站上用nginx做了一下反向代理,最近发现不能获取客户端ip了,都是拿到的127.0.0.1的本地ip... 通过查资料后,再去看了看我的配置文件,结果发现我没有如下配置: nginx ...

  4. spring beans

    所 有 使 用 XML 文 件 进 行 配 置 信 息 加 载 的 Spring IoC 容 器 , 包 括 BeanFactory 和ApplicationContext的所有XML相应实现,都使用 ...

  5. 框架 Onboard-引导页样式制作库

    设置背景图片或者背景movie,然后在它们之上生成数个ViewController,默认是顶部一张图片,下面是标题和详细介绍,最下面是按钮和pagegithub地址  https://github.c ...

  6. MySQL 性能优化 30个数据库设计的最佳实践

    数据库设计是整个程序的重点之一,为了支持相关程序运行,最佳的数据库设计往往不可能一蹴而就,只能反复探寻并逐步求精,这是一个复杂的过程,也是规划和结构化数据库中的数据对象以及这些数据对象之间关系的过程. ...

  7. delphi中线程应用之Synchronize

    当一个线程在使用时,如果这个函数使用了Synchronize修钸的话就不允许别一个线程来调用这个函数,它的目的是避免多个子线程同时访问主线程资源.示例:procedure TTaskThread.Sh ...

  8. 关于XE10下Indy发送字符串编码的问题

    在与硬件对接的过程中,之前用D7环境下的UDPServer.Post发送的指令,硬件可正常识别并正常显示, 后来使用到XE10,重新编译之前的源码,发现所有汉字乱码显示了: 后通过对接收数据发现,实际 ...

  9. leetcode6

    好吧,今天晚上赶项目确实是做不了三道题目了,最近项目在网络编程方面有些进步,学到了东西,有时间再积累下来,很深的体会就是,和别人一起写代码,虽然蛋疼但是比自己一个人写要好点,不过发现自己对链表和排序什 ...

  10. C#枚举中的位运算权限分配浅谈

    常用的位运算主要有与(&), 或(|)和非(~), 比如: 1 & 0 = 0, 1 | 0 = 1, ~1 = 0 在设计权限时, 我们可以把权限管理操作转换为C#位运算来处理. 第 ...