Android中的XML解析-DOM的使用与开发技巧
我们现在的大多开发项目中都难免和服务器端打交道,交互的过程无非是数据输入和输出,当我们将接口要求的数据按照格式传输到服务器端后,就会得到服务器端返回的数据,在行业内数据大多会使用被普遍接受和采用的XML格式(也有不少使用JSON)来组织数据进行返回,当我们获得XML时就可以解析获得自己需要的数据,或许是操作结果,或许是需要展示的数据列表,在Android中主要有三种的XML解析方式分别是SAX、DOM、PULL,这期我主要讲DOM的基本原理、应用和开发技巧,其他的两种方式会在后面的几期进行介绍。
1. DOM的基本原理:
DOM解析XML文件时,会将XML文件的所有内容读取到内存中,然后允许您使用DOM API遍历XML树、检索所需的数据。使用DOM操作XML的代码看起来比较直观,并且,在某些方面比基于SAX的实现更加简单。但是,因为DOM需要将XML文件的所有内容读取到内存中,所以内存的消耗比较大,特别对于运行Android的移动设备来说,因为设备的资源比较宝贵,所以建议还是采用SAX来解析XML文件,当然,如果XML文件的内容比较小采用DOM是可行的。
2. DOM的基本使用
<?xml version="1.0" encoding="UTF-8"?> <persons> <person id="1"> <name>叶问</name> <age>23</age> </person> <person id="2"> <name>李小龙</name> <age>17</age> </person> </persons>
import java.io.InputStream; import java.util.ArrayList; import java.util.List; import javax.xml.parsers.DocumentBuilder; import javax.xml.parsers.DocumentBuilderFactory; import org.w3c.dom.Document; import org.w3c.dom.Element; import org.w3c.dom.Node; import org.w3c.dom.NodeList; /** * 使用Dom解析xml文件 * */ public class DomXMLReader { public static List<Person> readXML(InputStream inStream) { List<Person> persons = new ArrayList<Person>(); DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance(); try { DocumentBuilder builder = factory.newDocumentBuilder(); Document dom = builder.parse(inStream); Element root = dom.getDocumentElement(); NodeList items = root.getElementsByTagName("person"); //查找所有person节点 for (int i = 0; i < items.getLength(); i++) { Person person = new Person(); //得到第一个person节点 Element personNode = (Element) items.item(i); //获取person节点的id属性值 person.setId(new Integer(personNode.getAttribute("id"))); //获取person节点下的所有子节点(标签之间的空白节点和name/age元素) NodeList childsNodes = personNode.getChildNodes(); for (int j = 0; j < childsNodes.getLength(); j++) { Node node = (Node) childsNodes.item(j); //判断是否为元素类型 if(node.getNodeType() == Node.ELEMENT_NODE){ Element childNode = (Element) node; //判断是否name元素 if ("name".equals(childNode.getNodeName())) { //获取name元素下Text节点,然后从Text节点获取数据 person.setName(childNode.getFirstChild().getNodeValue()); } else if (“age”.equals(childNode.getNodeName())) { person.setAge(new Short(childNode.getFirstChild().getNodeValue())); } } } persons.add(person); } inStream.close(); } catch (Exception e) { e.printStackTrace(); } return persons; }
3. Dom开发小技巧
1) 快速获得一个简单XML中的某项数据,改方法只适用于某个属性在DOM中只存在一个
public static String getValByTagName(Document doc, String tagName) { NodeList list = doc.getElementsByTagName(tagName); if (list.getLength() > 0) { Node node = list.item(0); Node valNode = node.getFirstChild(); if (valNode != null) { String val = valNode.getNodeValue(); return val; } } return null; }
2) 快速的将数据列表转换成对象列表
如上述基本使用中的XML例子,先建立一个和XML对应的类,属性名称必须与XML中的名称一致,如:
public class Person(){ private String name; private String age; public void setName(String name){ this.name = name; } public String getName(){ return name; } public void setAge(String age){ this.age = age; } public String getAge(){ return age; } }
然后采用映射机制
NodeList nodeList = doc.getElementsByTagName("person"); int len = nodeList.getLength(); for (int i = 0; i < len; i++) { Node node = nodeList.item(i); Person bean = (Person) convertToBean(Person.class, node); } public static Object convertToBean(Class clazz, Node ele) { Object object = null; try { object = clazz.newInstance(); Class[] parameterTypes = new Class[] { String.class }; NodeList list = ele.getChildNodes(); int len = list.getLength(); for (int i = 0; i < len; i++) { try { Node node = list.item(i); String nodeName = node.getNodeName(); String nodeVal = ""; Node valueNode = node.getFirstChild(); if (valueNode != null) { nodeVal = valueNode.getNodeValue(); } Object[] arguments = new Object[] { nodeVal }; if (nodeName.length() > 0) { String first = nodeName.substring(0, 1); String last = nodeName.substring(1); Method method = clazz.getMethod("set" + first.toUpperCase() + last, parameterTypes); method.invoke(object, arguments); } } catch (NoSuchMethodException e) { } } return object; } catch (Exception e) { e.printStackTrace(); return null; } }
这种方法可以应用于大多数的使用场景