今天晚上我学习了下XML的解析,解析XML方法有SAX和DOM解析和Pull解析,今天先学习了下SAX,每天8点下班,下班回来还要自己做饭,所以每天我都要奋斗到1点左右,然后早上8点起来做饭带到公司去吃,刚毕业的程序猿很苦逼吧,不过不要紧,我相信通过我的努力,这一切都会改变,fighting...哈哈,废话不多说了,看项目吧
如果要用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");
/** 坑爹的地方,当我调用getInputStreamContent (InputStream is)这个方法时,解析就错误,不知道为什么???*/
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();
}
}
我只是调用那个方法就错误
org.apache.harmony.xml.ExpatParser$ParseException: At line 1, column 0: no element found at org.apache.harmony.xml.ExpatParser.finish(ExpatParser.java:550) at org.apache.harmony.xml.ExpatParser.parseDocument(ExpatParser.java:480) at org.apache.harmony.xml.ExpatReader.parse(ExpatReader.java:318) at org.apache.harmony.xml.ExpatReader.parse(ExpatReader.java:275) at javax.xml.parsers.SAXParser.parse(SAXParser.java:390) at javax.xml.parsers.SAXParser.parse(SAXParser.java:187) at com.example.xml_parser.SAXforHandler.sax_XML(SAXforHandler.java:146) at com.example.xml_parser.TestForXML.testSax_XML(TestForXML.java:7) at java.lang.reflect.Method.invokeNative(Native Method) at android.test.AndroidTestRunner.runTest(AndroidTestRunner.java:169) at android.test.AndroidTestRunner.runTest(AndroidTestRunner.java:154) at android.test.InstrumentationTestRunner.onStart(InstrumentationTestRunner.java:529) at android.app.Instrumentation$InstrumentationThread.run(Instrumentation.java:1448)
那位大牛知道是什么情况,我又没有改变InputStream,为什么会出现异常,求大牛赐教,好了,洗澡睡觉去了,希望明天早上看到大牛为我解答