android之sax解析xml文件
2011-03-04 13:19
423 查看
beauties.xml
<?xml version="1.0" encoding="UTF-8"?> <beauties> <beauty> <name>林志玲</name> <age>28</age> </beauty> <beauty> <name>杨幂</name> <age>23</age> </beauty> </beauties>
main.xml
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:orientation="vertical" android:layout_width="fill_parent" android:layout_height="fill_parent" > <TextView android:id="@+id/textView" android:layout_width="fill_parent" android:layout_height="wrap_content" /> </LinearLayout>
activity的代码:
package cn.com.saxtest; import java.io.InputStream; import java.util.ArrayList; import javax.xml.parsers.SAXParser; import javax.xml.parsers.SAXParserFactory; import org.xml.sax.Attributes; import org.xml.sax.SAXException; import org.xml.sax.helpers.DefaultHandler; import android.app.Activity; import android.os.Bundle; import android.util.Log; import android.widget.TextView; /** * * @author chenzheng_java * @description 使用sax解析方式解析xml * @since 2010/03/04 * */ public class SaxParserActivity extends Activity { private ArrayList<Beauty> beautyList = new ArrayList<Beauty>(); private Beauty beauty = null; @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.main); SAXParserFactory factory = SAXParserFactory.newInstance(); try { SAXParser parser = factory.newSAXParser(); InputStream inputStream = this.getClassLoader() .getResourceAsStream("beauties.xml"); parser.parse(inputStream, new MyDefaultHandler()); String result = ""; for(Beauty beauty:beautyList){ result+="/n "+beauty.toString(); } TextView textView = (TextView) findViewById(R.id.textView); textView.setText(result); } catch (Exception e) { e.printStackTrace(); } } /** * * @author chenzheng_java * @description 一定要注意,这里是使用的内部类哦。 有些朋友可能有疑问,不用内部类不可以,答案是:当然可以。 * 这里用内部类的目的其实就是为了提高一丁点效率而已。 如果你喜欢看源码,你就会发现,源码中大量的使用了内部类的。 * */ private class MyDefaultHandler extends DefaultHandler { // 存储目前为止读取到的最后一个element的localname private String currentElementName = ""; /** * characters (char ch[], int start, int length)当解析xml中遇到文本内容时会执行。 ch * 这个数组中存放的是整个xml文件的字符串的数组形式 start是当前解析的文本在整个xml字符串文件中的开始位置 * length是当前解析的文本内容的长度 由上面的介绍我们可以知道,我们可以通过new * String(ch,start,length)方法来获取我们正解析的文本内容 */ @Override public void characters(char[] ch, int start, int length) throws SAXException { Log.i("currentElementName", currentElementName); String textContent = new String(ch, start, length); if(currentElementName.equals("name")&&textContent!=null&&!textContent.trim().equals("")){ Log.i("textContent name", textContent); beauty.setName(textContent); } if(currentElementName.equals("age")&&textContent!=null&&!textContent.trim().equals("")){ Log.i("textContent age", textContent); beauty.setAge(textContent); } } /** *解析到xml文档的末尾时触发 */ @Override public void endDocument() throws SAXException { } /** * 解析到元素的末尾时触发 */ @Override public void endElement(String uri, String localName, String qName) throws SAXException { if(localName.equals("beauty")){ beautyList.add(beauty); Log.i("beauty", beauty.toString()); } } /** * 开始解析xml时触发 */ @Override public void startDocument() throws SAXException { } /** * 解析到元素的开始处触发 startElement (String uri, String localName, String qName, * Attributes attributes) uri:Namespace值,当用户没有明确指定以及当命名空间没有被使用的时候,为null * localName:element的名称,或者通俗点叫标签的名称。如<name>中的name就是localName qName: * 和localName的唯一其别是 * ,当标签有namespace时,该值返回的数据为全限定名称。例如<chen:name>中,localName为name * ,qName为chen:name attributes:元素包含的属性对象。如果没有属性时,返回一个空的属性对象 */ @Override public void startElement(String uri, String localName, String qName, Attributes attributes) throws SAXException { currentElementName = localName; if(localName.equals("beauty")){ beauty = new Beauty(); } } } /** * * @author chenzheng 这里使用内部类是为了效率考虑,内部类要比单独顶一个bean类更加的高效以及节约空间 * */ private class Beauty { String name; String age; 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 "美女资料 [年龄=" + getAge() + ", 姓名=" + getName() + "]"; } } }
运行后就会得到想要的结果了、
![](http://hi.csdn.net/attachment/201103/4/0_12992160497Pvw.gif)
代码结构:
![](http://hi.csdn.net/attachment/201103/4/0_1299216070qqDc.gif)
----------------------------------------------------------------------------------------------------------
下面,再让我们一起讨论点废话:
SAX 解析
1. SAX ( Simple Application interface for XML ), 是一组程序设计接口,采用 observer 模式,将XML文件视为一个文字流的数据,在读取XML 元素时触发一系列的事件。这是观察者模式的一个典型应用。
2. 使用SAX 加载XML文件时,他的操作像打开一个“顺序的文件字符流”,在读到XML元素的开始标记,结尾标记和内容标记时将产生一系列的事件
如一个简单的XML文件:<hello><message>hello XML!</message></hello>
会相应的触发:startDocument, startElement, characters, endElement, endDocument, 只需编写这些事件处理程序就可以解析XML文件了
3. SAX 可以高效的使用内存,因为SAX 只是顺序的读取XML 文件的内容,并不会将XML 文件完全加载,这样就比DOM 的处理效率高
但SAX 只能读取XML 文件的内容,而不能更改XML 的内容,也不能随机访问XML 元素
4. 在SAX 中有4个处理器是要实现的:ContentHandler,DTDHandler,EntityResolver,ErrorHandler,以处理不同的事件,这是比较麻烦的,
幸好SAX 定义了一个 DefaultHandler 类把这几个实现了,我们只需在 DefaultHandler中定义事件处理方法,然后注册到XMLReader,而SAXParser封装了XMLReader的实现类,
SAXParser又是由SAXParserFactory提供的,所以我们实际用到的类只有:SAXParserFactory,SAXParser,DefaultHandler
5. SAX 的解析步骤:
(1)写一个类继承 DefaultHandler, 实现自己的事件处理方法
(2)在主程序中建立 SAXParserFactory
(3)可以设置这个factory 的参数
(4)从这个factory 得到SAXParser
(5)解析XML文件
相关文章推荐
- android 用 SAX 解析xml文件!
- Android之SAX解析XML文件
- android应用开发之利用SAX、DOM和Pull实现对XML文件的解析并进行单元测试
- android学习——使用SAX、DOM 和 PULL 解析xml文件,及使用pull生成xml文件
- Android学习札记10:用SAX解析XML文件时,当标签中出现换行符时可能出现读取内容为空的解决方法
- Android解析xml文件的方法:SAX
- android学习——使用SAX、DOM 和 PULL 解析xml文件,及使用pull生成xml文件
- android 用 SAX 解析xml文件!
- Android 解析XML文件的三种方式 DOM,SAX,PULL
- Android中利用SAX解析XML文件(转)
- 详解Android之解析XML文件三种方式(DOM,PULL,SAX)
- Android解析自定义xml文件--Sax解析xml文件,测试demo(方案二)
- android_xml解析之 使用SAX读取XML文件
- Android之SAX解析Xml文件实现
- Android中使用SAX方式解析XML文件
- android解析xml文件的方式(其二)SAX
- Android----XML文件解析之SAX模式解析
- android平台中解析xml文件:SAX、PULL
- Android 个人学习笔记之---SAX解析XML文件(有一个坑爹的问题)
- android如何使用DOM和SAXParserFactory来解析XML文件