之前对xml解析基本上不了解,学习之后,特写一篇文章用以记住解析方法,希望大家多多指教。
注:由于android本身已经集成了pull解析,所以在用的时候就不需要导入jar包,而j2ee开发时则需要导入相应的jar包。
=========================================================================
这是备用的xml文件命名为test.xml,放在src文件下,方便类加载器读取。
<?xml version="1.0" encoding="UTF-8"?>
<persons>
<person id="lm">
<name>liming</name>
<age>22</age>
</person>
<person id="zs">
<name>zhangsan</name>
<age>20</age>
</person>
</persons>
我们读取这个xml文件后,应该把它封装到一个实体类里边,所以我们新建一个Person类,应xml文件的属性,为了显示是否转换成功,我们再重写一些toString方法,显示属性,类如下:
public class Person {
private String id;
private String name;
private String age;
public Person() {
}
public Person(String id, String name, String age) {
this.id = id;
this.name = name;
this.age = age;
}
public String getId() {
return id;
}
public void setId(String id) {
this.id = id;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public String getAge() {
return age;
}
public void setAge(String age) {
this.age = age;
}
@Override
public String toString() {
return "Person [id=" + id + ", name=" + name + ", age=" + age + "]";
}
}
现在xml和实体类都准备好了,那么我们就可以开始写怎么转换的方法了,这样的业务逻辑,一般建议写在一个相关的Service里边,所以我们新建一个叫做PersonService的类,里边包含pull解析xml文件的方法,如下:
import java.io.InputStream;
import java.util.ArrayList;
import java.util.List;
import net.arvin.domain.Person;
import org.xmlpull.v1.XmlPullParser;
import org.xmlpull.v1.XmlPullParserFactory;
public class PersonService {
//我们想要解析xml文件,我们可以先将xml以输入流的方式读取出来,然后再处理,这个输入流的xml可以通过类加载器来读取,由于xml文件是放在src文件下的,所以方法如下:InputStream xml = this.getClass().getClassLoader().getResourceAsStream("test.xml");
public static List<Person> getPersons(InputStream xml) throws Exception {
List<Person> persons = null;
Person person = null;
XmlPullParser pullParser = XmlPullParserFactory.newInstance()
.newPullParser();
//XmlPullParser pullParser = Xml.newPullParser();<span style="font-family: Arial, Helvetica, sans-serif;">// 或者通过这种方法得到pull解析器
pullParser.setInput(xml, "UTF-8");// 为pull解析器设置要解析的xml数据
int event = pullParser.getEventType();
while (event != XmlPullParser.END_DOCUMENT) {
switch (event) {
case XmlPullParser.START_DOCUMENT:
persons = new ArrayList<Person>();
break;
case XmlPullParser.START_TAG://<span style="font-family: Arial, Helvetica, sans-serif;">解析到开始节点时会触发该事件,例如<persons><person><name><age>节点
if ("person".equals(pullParser.getName())) {//getName是获取当前节点的名字
person = new Person();
person.setId(pullParser.getAttributeValue(0));//getAttributeValue(index)获取当前节点的属性,因为person节点上属性只有id一个所以index = 0,若,id后编还有一个属性,比如叫做height,那么index = 1;
}
if ("name".equals(pullParser.getName())) {
person.setName(pullParser.nextText());//nexText()这个方法会获得当前节点的下一个文本节点
}
if ("age".equals(pullParser.getName())) {
person.setAge(pullParser.nextText());
}
break;
case XmlPullParser.END_TAG:<span style="font-family: Arial, Helvetica, sans-serif;">//解析到结束节点时会触发该事件,例如</persons></person></name></age>节点</span>
if ("person".equals(pullParser.getName())) {
persons.add(person);
person = null;
}
break;
}
event = pullParser.next();
}
return persons;
}
}
由于pull解析的原理是,读取到某一个节点,他就会触发相应的事件,所以,我们可以通过事件来捕获,pull解析器解析到哪里了。
当pull解析器,把<?xml version="1.0" encoding="UTF-8"?>读取完的时候,他会判断这是不是一个xml,如果是,那么它就会触发start document这个事件,此时我们可以做一些初始化的工作;pull解析有个特点,他自动读取到这之后他就不再自动解析,需要调用pullParse.next()方法才会继续读取下一个节点。由此可见我们应该在一个循环中不断读取,所以我们就需要有个循环结束的条件,当然就是文档读取完毕就循环结束,正好,文档读取完毕pull解析器会触发一个end document事件。
大体的代码就这么一点。希望大家多多指教。