SAX解析
如果要用SAX来解析xml文档,则需要一个类来继承android系统提供的ContentHandler类。但是如果继承ContentHandler这个类, 即使你不使用这个类提供的所有方法,你也必须实现其内部的所有方法(一般情况下没有使用的方法可以直接用空方法代替),但是这样开发起来不是很方便。因此我们可以改为继承DefaultHandler这个类,这样的话我们只需要实现程序中所需要的方法即可,其它的方法这个类内部其实已经用空方法代替了。
- ContentHandler接口的方法有以下几种:
- void startDocument();//文档解析开始时执行
- void endDocument();//文档解析结束时执行
- void startElement(String uri, String localName, String qName, Attributes atts);//标签开始解析时执行
- void endElement(String uri, String localName, String qName, Attributes atts);//标签解析结束时执行
- void characters(char[] ch, int start, int length );//解析标签属性时执行
android中使用SAX来解析xml文件,需先建立一个SAX工厂,即SAXParserFactory对象,还需建立一个XMLReader对象,该类绑定ContentHandler子类,且与xml源文件结合在一起。即其处理过程为创建事件处理程序,创建SAX解析器,键事件处理程序分配给解析器,对文档进行解析,将每个事件发送给处理程序。
1.在src目录下新建一个android.xml文件
- <?xml version="1.0" encoding="UTF-8"?>
- <persons>
- <person id="23">
- <name>xiaanming</name>
- <age>23</age>
- </person>
- <person id="20">
- <name>liudehua</name>
- <age>28</age>
- </person>
- </persons>
2.新建一个Person.class用来存放解析的对象
- package com.example.xml_parser;
- public class Person {
- private int id;
- private String name;
- private int age;
- public Person(){}
- public Person(int id, String name, int age){
- this.id = id;
- this.name = name;
- this.age = age;
- }
- 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 getAge() {
- return age;
- }
- public void setAge(int age) {
- this.age = age;
- }
- @Override
- public String toString() {
- return "id = " + id + ", name " + name + ", age = " + age;
- }
- }
3.新建一个SAXforHandler类继承DefaultHandler,而DefaultHandler实现了ContentHandler接口,需要重写我们需要的方法
- package com.example.xml_parser;
- import java.io.BufferedReader;
- import java.io.IOException;
- import java.io.InputStream;
- import java.io.InputStreamReader;
- import java.util.ArrayList;
- import java.util.List;
- import javax.xml.parsers.SAXParser;
- import javax.xml.parsers.SAXParserFactory;
- import org.xml.sax.Attributes;
- import org.xml.sax.InputSource;
- import org.xml.sax.SAXException;
- import org.xml.sax.XMLReader;
- import org.xml.sax.helpers.DefaultHandler;
- import android.util.Log;
- public class SAXforHandler extends DefaultHandler {
- private static final String TAG = "SAXforHandler";
- /**
- * 用来存放解析的Person对象
- */
- private List<Person> persons;
- /**
- * Person 对象的引用,记录当前的Person
- */
- private Person person;
- /**
- * 通过此变量,记录当前一个标签的名称
- */
- private String tag ;
- /**
- * 此方法只有在开始解析文档的时候执行一次,比较适合处理一些初始化的东西
- * 我new 了一个ArrayList<Person>()对象和打印Log
- */
- @Override
- public void startDocument() throws SAXException {
- persons = new ArrayList<Person>();
- Log.i(TAG, "****startDocument*****");
- }
- /**
- * 文档解析完了调用的回调方法
- */
- @Override
- public void endDocument() throws SAXException {
- Log.i(TAG, "****endDocument*****");
- }
- /**
- * uri 是命名空间
- * localName 标签的名称,如name, age 等
- * qName 带命名空间的标签名
- * Attributes 存放改标签的所有属性
- *
- * 当localName为person的时候, 我们拿出 person标签的属性值,此处由于只有一个属性,也可以直接person.setId(Integer.valueOf(attributes.getValue(0)));
- * 然后设置tag = person,这个方法执行完了,然后会执行回调方法characters(char[] ch, int start, int length),这是一个循环的过程
- */
- @Override
- public void startElement(String uri, String localName, String qName, Attributes attributes) throws SAXException {
- if("person".equals(localName)){
- for(int i=0; i<attributes.getLength(); i++){
- Log.i(TAG, "attributesName: " + attributes.getLocalName(i) + "__attributesValue: " + attributes.getValue(i) );
- person = new Person();
- person.setId(Integer.valueOf(attributes.getValue(i)));
- }
- }
- tag = localName;
- Log.i(TAG, "localName = " + localName);
- }
- /**
- * 这个方法只要是获取两个标签里面的值的,这里最好用trim()方法过滤下,可以避免读取到的XML有空格带来不必要的麻烦
- * 执行完这个方法就执行回调方法endElement(String uri, String localName, String qName),这也是一个循环的过程
- */
- @Override
- public void characters(char[] ch, int start, int length)
- throws SAXException {
- String date = new String(ch, start, length).trim();
- if(!"".equals(date)){
- Log.i(TAG, "Content: " + date);
- }
- if("name".equals(tag)){
- person.setName(date);
- }else if("age".equals(tag)){
- person.setAge(Integer.valueOf(date));
- }
- }
- /**
- * uri ,localName, qName跟上面的一个意思
- * 当localName = person 并且person对象为 null时 ,说明一个person对象解析完毕
- * 将person加入到List当中
- * 每个标签解析完了需要将tag = null
- */
- @Override
- public void endElement(String uri, String localName, String qName)
- throws SAXException {
- if("person".equals(localName) && person != null){
- persons.add(person);
- person = null;
- }
- tag = null;
- }
- /**
- * 拿到成员变量List<Person> persons的方法
- * @return
- */
- public List<Person> getPersons() {
- return persons;
- }
- /**
- * 1.加载需要解析的文件,因为XML放在src目录下,可以通过类装载器的方法获得文件路径,在以输入流的方式加入解析器
- * 2.解析XML有两种形式,创建一个XMLReader 或者直接使用XMLParser
- * @return
- * @throws Exception
- */
- public static List<Person> sax_XML() throws Exception{
- InputStream is = MainActivity.class.getClassLoader().getResourceAsStream("android.xml");
- <span style="font-size:32px;color:#ff0000;">/** 坑爹的地方,当我调用getInputStreamContent (InputStream is)这个方法时,解析就错误,不知道为什么???*/
- </span>
- SAXforHandler saXforHandler = new SAXforHandler();
- SAXParserFactory spf = SAXParserFactory.newInstance();
- SAXParser saxParser = spf.newSAXParser();
- //使用XMLReader的方式
- // XMLReader xmlReader = saxParser.getXMLReader();
- // xmlReader.setContentHandler(saXforHandler);
- // xmlReader.parse(new InputSource(is));
- //直接使用XMLParser,推荐使用这种
- saxParser.parse(is, saXforHandler);
- //获取解析好了的List对象
- List<Person> list = saXforHandler.getPersons();
- is.close();
- return list;
- }
- /**
- * 测试方法,根据输入流获取里面的内容
- * @param is
- * @return
- * @throws IOException
- */
- public static String getInputStreamContent (InputStream is) throws IOException{
- StringBuffer sb = new StringBuffer();
- BufferedReader br = new BufferedReader(new InputStreamReader(is));
- String line;
- while((line = br.readLine()) != null){
- sb.append(line);
- }
- br.close();
- Log.i(TAG, sb.toString());
- return sb.toString();
- }
- }
DOM解析
DOM比SAX更容易掌握,因为她没有涉及回调和复杂的状态管理,然而,DOM的实现常常将所有的XML节点保存在内存中,这样使的处理较大的文档效率低。
XML基本的节点类型
node - DOM基本的数据类型
Element - 最主要处理的对象是Element
Attr - 元素的属性
Text - 一个Element 或者Attr的实际内容
Document - 代表整个XML文档,一个Document对象通常也称为一颗DOM树
1 新建一个DomPersonService.class,注释我写的清楚,大家自己看
- package com.example.dom_parser;
- import java.io.InputStream;
- import java.util.ArrayList;
- import java.util.List;
- import javax.xml.parsers.DocumentBuilder;
- import javax.xml.parsers.DocumentBuilderFactory;
- import org.w3c.dom.Document;
- import org.w3c.dom.Element;
- import org.w3c.dom.Node;
- import org.w3c.dom.NodeList;
- import android.util.Log;
- public class DomPersonService {
- public static List<Person> readXML() throws Throwable{
- //获得android.xml文件的输入流
- InputStream is = MainActivity.class.getClassLoader().getResourceAsStream("android.xml");
- List<Person> persons = new ArrayList<Person>();
- //实例化DocumentBuilderFactory和DocumentBuilder,并创建Document
- DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance();
- DocumentBuilder builder = factory.newDocumentBuilder();
- Document document = builder.parse(is);
- //返回文档的根(root)元素
- Element rootElement = document.getDocumentElement();
- //获取一个Note(DOM基本的数据类型)集合,这里有两个person Note
- NodeList nodes = rootElement.getElementsByTagName("person");
- //遍历Note集合
- for(int i=0; i<nodes.getLength(); i++){
- //先从第一个person元素开始解析
- Element personElement = (Element) nodes.item(i);
- Person person = new Person();
- person.setId(Integer.valueOf(personElement.getAttribute("id")));
- //获取person下面的name 和 age 的Note集合
- NodeList chileNodes = personElement.getChildNodes();
- for(int y=0; y<chileNodes.getLength(); y++){
- Node childNode = chileNodes.item(y);
- //判断子Note的类型为元素Note
- if(childNode.getNodeType() == Node.ELEMENT_NODE){
- Element childElement = (Element) childNode;
- if("name".equals(childElement.getNodeName())){
- person.setName(childElement.getFirstChild().getNodeValue());
- }else if("age".equals(childElement.getNodeName())){
- person.setAge(Integer.valueOf(childElement.getFirstChild().getNodeValue()));
- }
- }
- }
- Log.e("log", person.toString());
- persons.add(person);
- }
- return persons;
- }
- }
关于DOM解析XML,我们要清楚的知道个节点之间的关系,才能更好的操作对象树,值得注意的是在建立Element时,要注意jar包的导入, 要选择org.w3c.dom.Element,而不是其他的包.
Pull解析
1.Pull简介
Pull解析器是Android系统内置的的,Pull解析器与SAX解析器类似,他提供了类似的事件,如开始元素和介绍元素的事件,使用parser.next()可以进入下一个元素并触发相应的事件,然后进行相应的处理,当元素开始解析时,调用perser.nextText()方法就可以获取到下一个Text类型元素的值。
2.pull特点
一.简单的结构,一个接口,一个另外,一个工厂组成了Pull解析器
二.简单易用,Pull解析器只有一个重要的方法next(),他被用来检索下一个事件,而他的事件也仅仅只有五个,START_DOCUMENT, START_TAG ,TEXT, END_TAG, END_DOCUMENT
三.最小的内存消耗,Pull解析器和SAX解析器一样,对内存的暂用少,但是SAX解析稍微有点繁琐,DOM很耗内存,所以Pull被推荐使用
3,示例 Pull解析XML
新建一个PullXMLService
- package com.example.pull_parser;
- import java.io.InputStream;
- import java.util.ArrayList;
- import java.util.List;
- import org.xmlpull.v1.XmlPullParser;
- import android.util.Log;
- import android.util.Xml;
- public class PullXMLService {
- public static List<Person> readXML() throws Exception{
- //获取src目录下面的android.xml文件的输入流
- InputStream is = PullXMLService.class.getClassLoader().getResourceAsStream("android.xml");
- //用来存放解析的Person对象
- List<Person> persons = null;
- //一个标记
- boolean flag = false;
- Person person = null;
- //实例化一个XmlPullParser对象
- XmlPullParser parser = Xml.newPullParser();
- //设置输入流和编码
- parser.setInput(is, "UTF-8");
- //触发了第一个事件,根据XML的语法,也就是从他开始了解文档
- int eventCode = parser.getEventType();
- //如果获得的事件码如果是文档的结束,那么解析结束
- while (eventCode != XmlPullParser.END_DOCUMENT) {
- switch(eventCode){
- case XmlPullParser.START_DOCUMENT:{
- //开始解析的时候我们一般做一些初始化的操作
- persons = new ArrayList<Person>();
- break;
- }
- case XmlPullParser.START_TAG:{
- //判断当前的元素是否是需要检索的元素
- if("person".equals(parser.getName())){
- flag = true;
- person = new Person();
- person.setId(Integer.valueOf(parser.getAttributeValue(0)));
- }
- if(flag){
- if("name".equals(parser.getName())){
- person.setName(parser.nextText());
- }else if("age".equals(parser.getName())){
- person.setAge(Integer.valueOf(parser.nextText()));
- }
- }
- break;
- }
- case XmlPullParser.END_TAG:{
- if("person".equals(parser.getName()) && person != null){
- flag = false;
- persons.add(person);
- Log.e("log", person.toString());
- person = null;
- }
- break;
- }
- }
- //这一步很重要,该方法返回一个事件码,也是触发下一个事件的方法
- eventCode = parser.next();
- }
- return persons;
- }
- }
好了,这样子就解析完了android.xml 是不是很方便很简单呢,比SAX和DOM要更容易理解。