SAX解析xml文件
优点:不用事先调入整个文档,占用资源少。尤其在嵌入式环境,如Android,极力推荐采用SAX进行解析。 缺点:不像DOM一样将文档树长期留驻在内存,数据不是长久的。事件过后,若没保存数据,那么数据就会丢失。 使用场合:机器有性能限制,尤其是在嵌入式环境。 SAX解析XML步骤 在Android中解析XML文主要有三种方式,分别为SimpleAPIforXML(SAX)、DocumentObjectModel(DOM)和Android附带的PULL解析器。其中SAX是一个解析速度非常快并且占用内存少的XML解析器,非常适合Android手机等移动设备。 SAX解析XML文件采用事件驱动的方式进行,也就是说,SAX是逐行扫描文件,遇到符合条件的设定条件后就会触发特定的事件,回调你写好的事件处理程序。使用SAX的优势在于其解析速度较快,占用内存较少(相对于DOM而言)。而且SAX在解析文件的过程中得到自己需要的信息后可以随时终止解析,并不一定要等文件全部解析完毕。凡事有利必有弊,其劣势在于SAX采用的是流式处理方式,当遇到某个标签的时候,它并不会记录下以前所遇到的标签,也就是说,在处理某个标签的时候,比如在startElement方法中,所能够得到的信息就是标签的名字和属性,至于标签内部的嵌套结构,上层标签、下层标签以及其兄弟节点的名称等等与其结构相关的信息都是不得而知的。实际上就是把XML文件的结构信息丢掉了,如果需要得到这些信息的话,只能你自己在程序里进行处理了。所以相对DOM而言,SAX处理XML文档没有DOM方便,SAX处理的过程相对DOM而言也比较复杂。 使用SAX解析XML文件一般有以下五个步骤: 1、创建一个SAXParserFactory对象(通过类名很容易得知它利用工厂方法模式实现的); 2、调用SAXParserFactory中的newSAXParser方法创建一个SAXParser对象; 3、然后在调用SAXParser中的getXMLReader方法获取一个XMLReader对象; 4、在XMLReader中注册事件处理接口,一般有ContentHandler、ErrorHandler、DTDHandler、EntityHandler四种; 5、调用XMLReader中的parse方法解析指定的XML字符串对象; 步骤四中提到的四个Handler是事件处理接口,SAX的事件处理函数就定义在这四个接口中,利用SAX解析XML文件需要重写接口中的方法。其中ContentHandler用来处理XML中的内容,ErrorHandler用来处理错误,DTDHandler用来处理DTD,EntityHandler用来处理XML文档中的实体;最常用的是ContentHandler这个接口,下面是该接口中的一些常用方法: startDocument() 当遇到文档的开头的时候,调用这个方法,可以在其中做一些预处理的工作。 endDocument() 和上面的方法相对应,当文档结束的时候,调用这个方法,可以在其中做一些善后的工作。 startElement(Stringuri,StringlocalName,StringqName,Attributesatts) 当读到一个开始标签的时候,会触发这个方法。uri是命名空间(通过xmlns声明),localName是不带命名空间前缀的标签名,qName是带命名空间前缀的标签名。通过atts可以得到所有的属性名和相应的值。注意,如果没有指定Namespace,则qName可能为空,当然不同的SAX实现会有所不同,比如在Android中qName为空,而J2SE中localName为空,所以想要总是得到标签名,就需要检查这两个参数的值了。 endElement(Stringuri,Stringname) 这个方法和上面的方法相对应,在遇到结束标签的时候,调用这个方法。 characters(char[]ch,intstart,intlength) 这个方法用来处理在XML文件中读到的内容,第一个参数为文件的字符串内容,后面两个参数是读到的字符串在这个数组中的起始位置和长度,使用newString(ch,start,length)就可以获取内容。 1.把解析的person.xml文件放在src目录下 <?xml version="1.0" encoding="UTF-8"?> <persons> <person id="1"> <name>olay</name> <age>23</age> </person> <person id="2"> <name>qiu</name> <age>21</age> </person> <person id="3"> <name>james</name> <age>23</age> </person> <person id="4"> <name>kobe</name> <age>21</age> </person> </persons>
2.针对从xml中获取的信息,需要对其新建一个Person类,存放相关信息
package cn.com.entity; public class Person { private Integer id; private String name; private int age; public Integer getId() { return id; } public void setId(Integer 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.新建一个继承DefaultHandler的类SAXFroHandler。DefaultHandler是实现了ContentHandler接口。ContentHandler提供了相应的事件方法(也是解析xml文件的触发事件)
package cn.com.service; import java.util.ArrayList; import java.util.List; import org.xml.sax.Attributes; import org.xml.sax.SAXException; import org.xml.sax.helpers.DefaultHandler; import android.util.Log; import cn.com.entity.Person; public class SAXForHandler extends DefaultHandler { private static final String TAG="SAXForHandler"; private List<Person> persons; private Person person ; //记录当前person private String perTag; //记录前一个标签的名称 public List<Person> getPersons(){ return persons; } //合适在此事件中触发初始化行为 @Override public void startDocument() throws SAXException { persons = new ArrayList<Person>(); Log.i(TAG,"****startDocument()****"); } @Override public void endDocument() throws SAXException { Log.i(TAG,"****endDocument()****"); } //开始解析每个元素时都会调用该方法 @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++){ person = new Person(); person.setId(Integer.parseInt(attributes.getValue(i))); Log.i(TAG,"attributeName:"+attributes.getLocalName(i)+",attributeValues:"+attributes.getValue(i)); } } perTag=localName; Log.i(TAG,"****startElement()****"); } //每个元素结束的时候都会调用该方法 @Override public void endElement(String uri,String qName) throws SAXException { if("person".equals(localName)){ persons.add(person); person = null; } perTag=null; Log.i(TAG,"****endElement()****"); } //解析到每个元素的内容时会调用此方法 @Override public void characters(char[] ch,int start,int length) throws SAXException { String data = new String(ch,length).trim(); if("name".equals(perTag)){ person.setName(data); }else if("age".equals(perTag)){ person.setAge(Integer.valueOf(data)); } if(!"".equals(data.trim())){ Log.i(TAG,"content:"+data.trim()); } } }
4.测试
package com.example.xml_parser; import java.io.InputStream; import java.util.ArrayList; import java.util.HashMap; import java.util.List; import java.util.Map; import javax.xml.parsers.ParserConfigurationException; import javax.xml.parsers.SAXParser; import javax.xml.parsers.SAXParserFactory; import org.xml.sax.SAXException; import cn.com.entity.Person; import cn.com.service.SAXForHandler; import android.os.Bundle; import android.app.Activity; import android.util.Log; import android.view.Menu; import android.view.View; import android.widget.ArrayAdapter; import android.widget.ListView; import android.widget.SimpleAdapter; public class MainActivity extends Activity { private ListView listView = null; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); listView = (ListView) findViewById(R.id.listView1); } public void onClick(View view) throws Exception{ List<Person> persons=readXml() ; List<Map<String,String>> data = new ArrayList<Map<String,String>>(); for(Person person:persons){ Map<String,String> map = new HashMap<String,String>(); map.put("name",person.getName()); map.put("age",""+person.getAge()); data.add(map); } SimpleAdapter adapter = new SimpleAdapter(MainActivity.this,data,R.layout.person_main,new String[]{"name","age"},new int[]{R.id.name,R.id.age}); listView.setAdapter(adapter); } //调用sax解析xml的方法 public List<Person> readXml() throws Exception{ //加载需要解析的文件。因为XML文件放在src目录下,所以通过类装载器的方式获得文件路径,再以输入流的方式放入解析器 //SAX是以流的方式接受指定的xml内容,所以可以直接获取文件的输入流 InputStream is=this.getClass().getClassLoader().getResourceAsStream("person.xml"); //加载自己创建的处理类对象,并通过SAXParserFactory来创建具体的SAXParser工厂模式。 SAXForHandler saxForHandler = new SAXForHandler(); SAXParserFactory spf = SAXParserFactory.newInstance(); SAXParser saxParser = spf.newSAXParser(); saxParser.parse(is,saxForHandler); List<Person> persons=saxForHandler.getPersons(); is.close(); return persons; } }
布局文件
Activity_main.xml
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="fill_parent" android:layout_height="fill_parent" android:orientation="vertical" > <Button android:layout_height="wrap_content" android:layout_width="fill_parent" android:text="SAX解析XML" android:onClick="onClick" /> <ListView android:id="@+id/listView1" android:layout_width="match_parent" android:layout_height="wrap_content" > </ListView> </LinearLayout>
Person_main.xml
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="fill_parent" android:layout_height="fill_parent" > <TextView android:id="@+id/textView1" android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="姓名:" /> <TextView android:id="@+id/name" android:layout_width="wrap_content" android:layout_height="wrap_content" /> <View android:layout_width="25dp" android:layout_height="wrap_content" /> <TextView android:id="@+id/textView2" android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="年龄:" /> <TextView android:id="@+id/age" android:layout_width="wrap_content" android:layout_height="wrap_content" /> </LinearLayout>
效果: (编辑:李大同) 【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容! |