• Android中用PULL解析XML


       解析XML的方式有DOM,SAX,PULL,那为什么要在Android中使用PULL解析呢?首先来说一下PULL解析的优点,然后再说一下其他两种解析方式的缺点,答案就清晰可见啦。

      DOM不适合文档较大,内存较小的场景。DOM是基于树形结构的的节点或信息片段的集合,允许开发人员使用DOM API遍历XML树、检索所需数据。分析该结构通常需要加载整个文档和构造树形结构,然后才可以检索和更新节点信息。由于DOM在内存中以树形结构存放,因此检索和更新效率会更高。但是对于特别大的文档,解析和加载整个文档将会很耗资源。

      SAX解析器操作太笨重,SAX(Simple API for XML)解析器是一种基于事件的解析器,它的核心是事件处理模式,主要是围绕着事件源以及事件处理器来工作的。当事件源产生事件后,调用事件处理器相应的处理方法,一个事件就可以得到处理。在事件源调用事件处理器中特定方法的时候,还要传递给事件处理器相应事件的状态信息,这样事件处理器才能够根据提供的事件信息来决定自己的行为。SAX解析器的优点是解析速度快,占用内存少。比较适合在Android移动设备中使用。

      PULL解析的优点:①轻巧灵活,速度快。②占用内存下(很大的优势)③使用方便顺手。PULL解析器的运行方式和SAX类似,都是基于事件的模式。不同的是,在PULL解析过程中,我们需要自己获取产生的事件然后做相应的操作,而不像SAX那样由处理器触发一种事件的方法,执行我们的代码。PULL解析器小巧轻便,解析速度快,简单易用,非常适合在Android移动设备中使用,Android系统内部在解析各种XML时也是用PULL解析器。

      那我接下来就介绍下PULL解析器:

    首先在项目的assets目录中放置一个你要解析的XML文档person.xml,内容如下:

    <?xml version="1.0" encoding="UTF-8"?>
    <persons>
           <person id="1">
                 <name>john</name>
                 <age>23</age>
           </person>
           <person id="2">
                 <name>david</name>
                 <age>233</age>
           </person>
    </persons>

    然后根据xml中person对象建模,Person.java如下:

    package com.example.xml_paser;
    public class Person {
        private int id;
        private String name;
        private int age;
        public int getId() {
            return id;
        }
        public void setId(int 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;
        }
    }

    再使用PULLParser解析,代码如下:

    package com.example.xml_paser;

    import java.io.InputStream;
    import java.util.ArrayList;
    import java.util.List;

    import org.xmlpull.v1.XmlPullParser;

    import android.content.Context;
    import android.util.Log;
    import android.util.Xml;

    public class xml_paser {
        static final String TAG = "XMLPaser";
        private static final String ns = null;
        private Context context;

        public xml_paser(Context applicationContext) {
            // TODO Auto-generated constructor stub
            this.context = applicationContext;
        }

        public List<Person> parse(String xmlPath) {
            List<Person> psonList = new ArrayList<Person>();//创建person对象列表
            Person person = null;
            InputStream stream = null;
            // get xml parser
            XmlPullParser xmlParse = Xml.newPullParser();//创建XmlPullparser实例
            try {
                // get file stream and set encoding
                stream = this.context.getResources().getAssets().open(xmlPath);//初始化输入流
                xmlParse.setInput(stream, "utf-8");//设置输入流 并指明编码方式
                // get event type
                int evnType = xmlParse.getEventType();
                // continue to end document
                while (evnType != XmlPullParser.END_DOCUMENT) {
                    switch (evnType) {
                    case XmlPullParser.START_TAG:
                        String tag = xmlParse.getName();//获得解析标记名
                        if (tag.equalsIgnoreCase("person")) {//标记是person的情况
                            person = new Person();//创建person对象
                            person.setId(Integer.parseInt(xmlParse
                                    .getAttributeValue(ns, "id")));//获得输入流中的id属性,赋值给新person对象的id
                        } else if (person != null) {
                            // parse after tag
                            if (tag.equalsIgnoreCase("name")) {
                                person.setName(xmlParse.nextText());//获得输入流中的name属性,赋值给新person对象的name
                            } else if (tag.equalsIgnoreCase("age")) {
                                person.setId(Integer.parseInt(xmlParse.nextText()));
                            }
                        }
                        break;
                    case XmlPullParser.END_TAG:
                        if (xmlParse.getName().equalsIgnoreCase("person")
                                && person != null) {
                            psonList.add(person);//将person对象加入personlist中
                            person = null;
                        }
                        break;
                    default:
                        break;
                    }
                    evnType = xmlParse.next();
                }
            } catch (Exception e) {
                Log.d(TAG, e.toString());
            }
            return psonList;
        }
    }

    最后,我们需要调用PULL解析程序,这个步骤在MainActivity中完成:

    package com.example.xml_paser;
    import java.util.List;
    import android.app.Activity;
    import android.os.Bundle;
    import android.util.Log;
    import android.view.View;
    import android.widget.Button;
    public class MainActivity extends Activity {
        private Button btn_paserButton;
        private String urlString="person.xml";
        private xml_paser xml_paser;
        @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.activity_main);
            btn_paserButton=(Button)findViewById(R.id.button1);
            xml_paser=new xml_paser(getApplicationContext());
            btn_paserButton.setOnClickListener(new View.OnClickListener()
            {
                @Override
                public void onClick(View arg0) {
                    // TODO Auto-generated method stub
                    List<Person> list_per = xml_paser.parse(urlString);
                    Log.v("bbb", list_per.size()+"");
                    for (int i = 0; i < list_per.size(); i++)
                    {
                        //Log.v("aaa",list_per.get(i));
                        Log.v("aaa", list_per.get(i).getName().toString()+"");
                    }
                }
            });
        }
    }

    运行结果显示person的名字,结果如下:

    EB%O7W9J%~B{JY2S@ODWCIP

  • 相关阅读:
    【Spring源码深度解析系列 】Spring整体架构
    【Spring Boot && Spring Cloud系列】在spring-data-Redis中如何使用切换库
    自定义异常
    Java IO
    多线程和虚拟机的宏观理解
    mybatis添加记录时返回主键id
    从JVM角度看i++ 与++i
    JVM指令集介绍
    jvm 指令集代码
    javap -c命令详解
  • 原文地址:https://www.cnblogs.com/wucaiyun1/p/4715113.html
Copyright © 2020-2023  润新知