android之sax解析xml文件
16lz
2022-04-18
android之sax解析xml文件
beauties.xml
[xhtml] view plain copy- <?xmlversion="1.0"encoding="UTF-8"?>
- <beauties>
- <beauty>
- <name>林志玲</name>
- <age>28</age>
- </beauty>
- <beauty>
- <name>杨幂</name>
- <age>23</age>
- </beauty>
- </beauties>
main.xml
[xhtml] view plain copy
- <?xmlversion="1.0"encoding="utf-8"?>
- <LinearLayoutxmlns: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的代码:
[java] view plain copy- packagecn.com.saxtest;
- importjava.io.InputStream;
- importjava.util.ArrayList;
- importjavax.xml.parsers.SAXParser;
- importjavax.xml.parsers.SAXParserFactory;
- importorg.xml.sax.Attributes;
- importorg.xml.sax.SAXException;
- importorg.xml.sax.helpers.DefaultHandler;
- importandroid.app.Activity;
- importandroid.os.Bundle;
- importandroid.util.Log;
- importandroid.widget.TextView;
- /**
- *
- *@authorchenzheng_java
- *@description使用sax解析方式解析xml
- *@since2010/03/04
- *
- */
- publicclassSaxParserActivityextendsActivity{
- privateArrayList<Beauty>beautyList=newArrayList<Beauty>();
- privateBeautybeauty=null;
- @Override
- publicvoidonCreate(BundlesavedInstanceState){
- super.onCreate(savedInstanceState);
- setContentView(R.layout.main);
- SAXParserFactoryfactory=SAXParserFactory.newInstance();
- try{
- SAXParserparser=factory.newSAXParser();
- InputStreaminputStream=this.getClassLoader()
- .getResourceAsStream("beauties.xml");
- parser.parse(inputStream,newMyDefaultHandler());
- Stringresult="";
- for(Beautybeauty:beautyList){
- result+="/n"+beauty.toString();
- }
- TextViewtextView=(TextView)findViewById(R.id.textView);
- textView.setText(result);
- }catch(Exceptione){
- e.printStackTrace();
- }
- }
- /**
- *
- *@authorchenzheng_java
- *@description一定要注意,这里是使用的内部类哦。有些朋友可能有疑问,不用内部类不可以,答案是:当然可以。
- *这里用内部类的目的其实就是为了提高一丁点效率而已。如果你喜欢看源码,你就会发现,源码中大量的使用了内部类的。
- *
- */
- privateclassMyDefaultHandlerextendsDefaultHandler{
- //存储目前为止读取到的最后一个element的localname
- privateStringcurrentElementName="";
- /**
- *characters(charch[],intstart,intlength)当解析xml中遇到文本内容时会执行。ch
- *这个数组中存放的是整个xml文件的字符串的数组形式start是当前解析的文本在整个xml字符串文件中的开始位置
- *length是当前解析的文本内容的长度由上面的介绍我们可以知道,我们可以通过new
- *String(ch,start,length)方法来获取我们正解析的文本内容
- */
- @Override
- publicvoidcharacters(char[]ch,intstart,intlength)
- throwsSAXException{
- Log.i("currentElementName",currentElementName);
- StringtextContent=newString(ch,start,length);
- if(currentElementName.equals("name")&&textContent!=null&&!textContent.trim().equals("")){
- Log.i("textContentname",textContent);
- beauty.setName(textContent);
- }
- if(currentElementName.equals("age")&&textContent!=null&&!textContent.trim().equals("")){
- Log.i("textContentage",textContent);
- beauty.setAge(textContent);
- }
- }
- /**
- *解析到xml文档的末尾时触发
- */
- @Override
- publicvoidendDocument()throwsSAXException{
- }
- /**
- *解析到元素的末尾时触发
- */
- @Override
- publicvoidendElement(Stringuri,StringlocalName,StringqName)
- throwsSAXException{
- if(localName.equals("beauty")){
- beautyList.add(beauty);
- Log.i("beauty",beauty.toString());
- }
- }
- /**
- *开始解析xml时触发
- */
- @Override
- publicvoidstartDocument()throwsSAXException{
- }
- /**
- *解析到元素的开始处触发startElement(Stringuri,StringlocalName,StringqName,
- *Attributesattributes)uri:Namespace值,当用户没有明确指定以及当命名空间没有被使用的时候,为null
- *localName:element的名称,或者通俗点叫标签的名称。如<name>中的name就是localNameqName:
- *和localName的唯一其别是
- *,当标签有namespace时,该值返回的数据为全限定名称。例如<chen:name>中,localName为name
- *,qName为chen:nameattributes:元素包含的属性对象。如果没有属性时,返回一个空的属性对象
- */
- @Override
- publicvoidstartElement(Stringuri,StringlocalName,StringqName,
- Attributesattributes)throwsSAXException{
- currentElementName=localName;
- if(localName.equals("beauty")){
- beauty=newBeauty();
- }
- }
- }
- /**
- *
- *@authorchenzheng这里使用内部类是为了效率考虑,内部类要比单独顶一个bean类更加的高效以及节约空间
- *
- */
- privateclassBeauty{
- Stringname;
- Stringage;
- publicStringgetName(){
- returnname;
- }
- publicvoidsetName(Stringname){
- this.name=name;
- }
- publicStringgetAge(){
- returnage;
- }
- publicvoidsetAge(Stringage){
- this.age=age;
- }
- @Override
- publicStringtoString(){
- 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文件
更多相关文章
- android 报错问题解析
- 49 Android(安卓)fragment 自定义适配器adapter
- Android文件操作
- java.lang.NoSuchFieldError: com.banwxf.shoplibrary.R$id.page
- android 文件保存方法 sd卡中或系统
- [Android]关于filed 遍历资源文件的排序问题
- android 文件上传
- android一种较为复杂的布局参考(xml文件)
- NPM 和webpack 的基础使用