Android之DOM解析XML
一.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