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() + "]"; } } }

运行后就会得到想要的结果了、

代码结构:

----------------------------------------------------------------------------------------------------------

下面,再让我们一起讨论点废话:

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文件

更多相关文章

  1. android 将bitmap缓存到本地
  2. Android文件操作
  3. android 判断文件是否存在
  4. Android(安卓)使用decodeFile方法加载手机磁盘中的图片文件
  5. Android(安卓)读写XML文件(使用pull解析)
  6. Android作为HTTP服务器--NanoHTTPD源码分析
  7. 录音及播放音频文件
  8. android文件解析InputStream问题解决
  9. NPM 和webpack 的基础使用

随机推荐

  1. Android(安卓)service启动流程分析.
  2. android 左右滑动+索引图标实现方法与代
  3. Android(安卓)漂亮的日期选择控件Time-Se
  4. Android: Keyguard分析之一:开机流程篇
  5. 编写android加载图片的程序时,遇到了内存
  6. Android(安卓)ContentObserver实现数据库
  7. Android(安卓)Activity启动模式全面解析
  8. android 第三方app截图
  9. Android位图
  10. Activity屏幕旋转时被销毁