Android之DOM解析XML

手机APP开发 2016-12-15

一.DOM解析方法介绍

DOM是基于树形结构的节点或信息片段的集合,允许开发人员使用DOM API遍历XML树,检索所需数据。分析该结构通常需要加载整个文档和构造树形结构,然后才可以检索和更新节点信息。

Android完全支持DOM解析。利用DOM中的对象可以对XML文档进行读取、搜索、修改、添加和删除等操作。

DOM的工作原理:使用DOM对XML文件进行操作时,首先要解析文件,将文件分为独立的元素、树形和注释等,然后以节点树的形式在内存中对XML文件进行表示,就可以通过节点树访问文档的内容,并根据需要修改文档—这就是DOM的工作原理。

DOM实现时首先为XML文档的解析定义一组接口,解析器读入整个文档,然后构造一个驻留内存的树结构这样代码就可以使用DOM接口来操作整个树结构。

由于DOM在内存中以树形结构存放,因此检索和更新效率会更高。但是对于特别大的文档,解析和加载整个文档就回很耗资源。当然,如果XML文档的内容比较小,采用DOM是可行的。

XML基本的节点类型:

node---DOM基本的数据类型

Element---最主要的处理的对象是Element

Attr---元素的属性

Text---一个Element或者Attr的实际内容

Document---一个代表整个XML文档,一个Document对象通常也称为一个树。

优点:整个文档读入内存,方便操作,支持修改、删除和重新排列等多种功能。

缺点: 将整个文档读入内存中,保留了过多的不需要的节点,浪费内存和空间。

使用场合: 一旦读入文档,还需要多次对文档进行操作,并且在硬件资源充足的情况下(内存、CPU)。

二、DOM解析步骤

         1. 首先利用DocumentBuilderFactory创建一个DocumentBuilderFactory实例。

DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance();

         2. 利用DocumentBuilderFactory创建DocumentBuilder。

DocumentBuilder builder = factory.newDocumentBuilder();

         3. 加载XML文档。

Document dom = builder.parse(inputStream);

         4. 获取文档的根节点(Element)

Element root = dom.getDocumentElement();

         5. 获取根节点中的所有子节点的列表(NodeList)

NodeList items = root.getElementsByTagName("person");

         6. 然后再获取子节点列表中的需要读取的节点。

//所有子节点for (int i = 0; i < items.getLength(); i++) {
    //得到一个Person节点
    Element personNode = (Element) items.item(i);
    //获取Person节点下的所有子节点(标签之间的空白节点和name/age元素)
    NodeList childNodes = personNode.getChildNodes();   //子节点
    for (int j = 0; j < childNodes.getLength(); j++) {
        Node node = childNodes.item(j);     //判断是否为元素类型
        if (node.getNodeType() == Node.ELEMENT_NODE) {
             Element childNode = (Element) node;
             ...
         }
    }
}

三、DOM解析XML代码

1.创建一个XML文件itcase.xml,并将其放在res/raw文件夹下。
<?xml version="1.0" encoding="UTF-8"?>
<persons>
    <person id="23">
        <name>liming</name>
        <age>30</age>
    </person>
    <person id="20">
        <name>lixiangmei</name>
        <age>25</age>
    </person>
</persons>

如果没有raw文件夹,就在res文件夹下创建一个raw文件夹,并创建xml文件。

2.修改视图
<Button
        android:id="@+id/dom_button"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_centerHorizontal="true"
        android:layout_marginTop="@dimen/fab_margin"
        android:gravity="center_horizontal"
        android:text="@string/DOM" />
<TextView
        android:id="@+id/text"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="Hello World!" />
3.添加AnalyzeDOM类
package com.zhangmiao.analyzexmldemo;

import org.w3c.dom.Document;
import org.w3c.dom.Element;
import org.w3c.dom.Node;
import org.w3c.dom.NodeList;

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

import javax.xml.parsers.DocumentBuilder;
import javax.xml.parsers.DocumentBuilderFactory;

/**
 * Created by zhangmiao on 2016/12/14.
 */
public class AnalyzeDOM {

    public static List<Person> readXML(InputStream inputStream) {
        List<Person> persons = new ArrayList<>();
        DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance();
        try {
            DocumentBuilder builder = factory.newDocumentBuilder();
            Document dom = builder.parse(inputStream);
            Element root = dom.getDocumentElement();
            NodeList items = root.getElementsByTagName("person");
            for (int i = 0; i < items.getLength(); i++) {
                Person person = new Person();
                Element personNode = (Element) items.item(i);
                person.setId(new Integer(personNode.getAttribute("id")));
                NodeList childNodes = personNode.getChildNodes();
                for (int j = 0; j < childNodes.getLength(); j++) {
                    Node node = childNodes.item(j);
                    if (node.getNodeType() == Node.ELEMENT_NODE) {
                        Element childNode = (Element) node;
                        if ("name".equals(childNode.getNodeName())) {
                            person.setName(childNode.getFirstChild().getNodeValue());
                        } else if ("age".equals(childNode.getNodeName())) {
                            person.setAge(new Short(childNode.getFirstChild().getNodeValue()));
                        }
                    }
                }
                persons.add(person);
            }
            inputStream.close();
        } catch (Exception e) {
            e.printStackTrace();
        }
        return persons;
    }
}
4.修改MainActivity类
package com.zhangmiao.analyzexmldemo;

import android.os.Bundle;
import android.support.design.widget.FloatingActionButton;
import android.support.design.widget.Snackbar;
import android.support.v7.app.AppCompatActivity;
import android.support.v7.widget.Toolbar;
import android.util.Log;
import android.view.View;
import android.view.Menu;
import android.view.MenuItem;
import android.widget.Button;
import android.widget.TextView;

import org.xml.sax.InputSource;

import java.io.BufferedWriter;
import java.io.File;
import java.io.FileOutputStream;
import java.io.InputStream;
import java.io.OutputStreamWriter;
import java.io.StringWriter;
import java.util.List;

import javax.xml.parsers.SAXParser;
import javax.xml.parsers.SAXParserFactory;

public class MainActivity extends AppCompatActivity implements View.OnClickListener {

    private static final String TAG = "AnalyzeXMLDemo";

    private TextView mTextView;

    private InputStream inputStream;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        Log.v(TAG, "onCreate");
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        Toolbar toolbar = (Toolbar) findViewById(R.id.toolbar);
        setSupportActionBar(toolbar);

        FloatingActionButton fab = (FloatingActionButton) findViewById(R.id.fab);
        fab.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
                Snackbar.make(view, "Replace with your own action", Snackbar.LENGTH_LONG)
                        .setAction("Action", null).show();
            }
        });
        Button domButton = (Button) findViewById(R.id.dom_button);
        mTextView = (TextView) findViewById(R.id.text);

        domButton.setOnClickListener(this);
    }

    @Override
    public void onClick(View v) {
        String result = "";
        inputStream = getResources().openRawResource(R.raw.itcase);
        switch (v.getId()) {case R.id.dom_button:
                result += "--------- DOM ---------" + "\n";
                if (inputStream == null) {
                    result = "inputStream is null";
                } else {
                    List<Person> personList = AnalyzeDOM.readXML(inputStream);
                    if (personList != null) {
                        for (int i = 0; i < personList.size(); i++) {
                            String message = "id = " + personList.get(i).getId() + " , name = " + personList.get(i).getName()
                                    + " , age = " + personList.get(i).getAge() + ".\n";
                            result += message;
                        }
                    }
                }
                mTextView.setText(result);
                break;default:
                break;
        }
    }

    @Override
    public boolean onCreateOptionsMenu(Menu menu) {
        // Inflate the menu; this adds items to the action bar if it is present.
        getMenuInflater().inflate(R.menu.menu_main, menu);
        return true;
    }

    @Override
    public boolean onOptionsItemSelected(MenuItem item) {
        // Handle action bar item clicks here. The action bar will
        // automatically handle clicks on the Home/Up button, so long
        // as you specify a parent activity in AndroidManifest.xml.
        int id = item.getItemId();

        //noinspection SimplifiableIfStatement
        if (id == R.id.action_settings) {
            return true;
        }

        return super.onOptionsItemSelected(item);
    }
}

参考文章:

http://www.open-open.com/lib/view/open1392780226397.html

http://www.cnblogs.com/weixing/archive/2013/08/07/3243366.html

http://www.tuicool.com/articles/IvQvyq

相关推荐