android中数据存储及对xml的解析

XML作为一种业界公认的数据交换格式,在各个平台与语言之上,都有广泛使用和实现。

其标准型,可靠性,安全性......毋庸置疑。在android平台上,我们要想实现数据存储和数据交换,经常会使用到xml数据格式和xml文件。

小提示:android中存储数据一般有如下几种:SharedPreferences(参数化),XML文件,sqllite数据库,网络,ContentProvider(内容提供者)等。

在android中,操作xml文件,一般有几种方式:SAX操作,Pull操作,DOM操作等。其中DOM的方式,可能是大家最熟悉的,也是符合W3C标准的。

1)

在java平台中,有诸如DOM4J这样优秀的开源包,极大程度的方便大家使用DOM标准来操作XML文件。在javascript中,不同的浏览器解析引擎,对DOM的解析和操作也略有差异(不过这不是本章介绍的重点)。而DOM的方式,也有其缺点。通常一次性加载xml文件,再使用DOM的 api去进行解析,这样很大程度的消耗内存,对性能会有一定影响。而我们的android手机,虽然配置在不断的升级,但是内存方面,暂时还无法与传统的PC去媲美。所以,在android开发中,个人不太推荐使用DOM的方式来解析和操作XML。

[java]  view plain copy
  1. package cn.itcast.service;  
  2.   
  3. import java.io.InputStream;  
  4. import java.util.ArrayList;  
  5. import java.util.List;  
  6.   
  7. import javax.xml.parsers.DocumentBuilder;  
  8. import javax.xml.parsers.DocumentBuilderFactory;  
  9.   
  10. import org.w3c.dom.Document;  
  11. import org.w3c.dom.Element;  
  12. import org.w3c.dom.Node;  
  13. import org.w3c.dom.NodeList;  
  14.   
  15. import cn.itcast.model.Person;  
  16.   
  17. public class DomPersonService {  
  18.   
  19.      public List<Person> getPersons(InputStream stream) throws Throwable  
  20.      {  
  21.          List<Person> list =new ArrayList<Person>();  
  22.          DocumentBuilderFactory factory =DocumentBuilderFactory.newInstance();  
  23.          DocumentBuilder builder =factory.newDocumentBuilder();  
  24.         Document dom = builder.parse(stream);//解析完成,并以dom树的方式存放在内存中。比较消耗性能  
  25.          //开始使用dom的api去解析  
  26.          Element root = dom.getDocumentElement();//根元素  
  27.         NodeList personNodes = root.getElementsByTagName("person");//返回所有的person元素节点  
  28.         //开始遍历啦  
  29.         for(int i=0;i<personNodes.getLength();i++)  
  30.         {  
  31.             Person person =new Person();  
  32.         Element personElement =(Element)personNodes.item(i);  
  33.           person.setId(new Integer( personElement.getAttribute("id")));//将person元素节点的属性节点id的值,赋给person对象  
  34.           NodeList personChildrenNodes =personElement.getChildNodes();//获取person节点的所有子节点  
  35.           //遍历所有子节点  
  36.           for(int j=0;j<personChildrenNodes.getLength();j++)  
  37.           {  
  38.               //判断子节点是否是元素节点(如果是文本节点,可能是空白文本,不处理)  
  39.               if(personChildrenNodes.item(j).getNodeType()==Node.ELEMENT_NODE)  
  40.               {  
  41.                   //子节点--元素节点  
  42.                   Element childNode =(Element)personChildrenNodes.item(j);  
  43.                   if("name".equals(childNode.getNodeName()))  
  44.                   {  
  45.                       //如果子节点的名称是“name”.将子元素节点的第一个子节点的值赋给person对象  
  46.                       person.setName(childNode.getFirstChild().getNodeValue());  
  47.                         
  48.                   }else if("age".equals(childNode.getNodeValue()))  
  49.                   {   
  50.                       person.setAge(new Integer(childNode.getFirstChild().getNodeValue()));  
  51.                   }  
  52.               }  
  53.                 
  54.           }  
  55.           list.add(person);  
  56.         }  
  57.         return list;  
  58.      }  
  59. }  

2)

SAX(Simple API for XML),是一个使用非常广泛的XML解析标准,通常使用Handler模式来处理XML文档,这种处理模式和我们平常习惯的理解方式很不同,身边也经常有一些朋友在刚接触SAX的时候会觉得理解起来有些困难。其实SAX并不复杂,只不过是换了一种思维方式,正如它的名字所表示的,为了让我们以更简单的方式来处理XML文档,下面我们就开始吧。

[java]  view plain copy
  1. package cn.itcast.service;  
  2.   
  3. import java.io.InputStream;  
  4. import java.util.ArrayList;  
  5. import java.util.List;  
  6.   
  7. import javax.xml.parsers.SAXParser;  
  8. import javax.xml.parsers.SAXParserFactory;  
  9.   
  10. import org.xml.sax.Attributes;  
  11. import org.xml.sax.SAXException;  
  12. import org.xml.sax.helpers.DefaultHandler;  
  13.   
  14. import cn.itcast.model.Person;  
  15.   
  16. public class SAXPersonService {  
  17.    public List<Person> getPersons(InputStream inStream) throws Throwable  
  18.    {  
  19.        SAXParserFactory factory = SAXParserFactory.newInstance();//工厂模式还是单例模式?  
  20.        SAXParser parser =factory.newSAXParser();  
  21.        PersonParse personParser =new PersonParse();  
  22.        parser.parse(inStream, personParser);  
  23.        inStream.close();  
  24.        return personParser.getPerson();  
  25.    }  
  26.    private final class PersonParse extends DefaultHandler  
  27.    {  
  28.      
  29.       
  30.     private List<Person> list = null;  
  31.     Person person =null;  
  32.     private String tag=null;  
  33.       
  34.     public List<Person> getPerson() {  
  35.         return list;  
  36.     }  
  37.     @Override  
  38.     public void startDocument() throws SAXException {  
  39.         list =new ArrayList<Person>();  
  40.     }  
  41.   
  42.     @Override  
  43.     public void startElement(String uri, String localName, String qName,  
  44.             Attributes attributes) throws SAXException {  
  45.         if("person".equals(localName))  
  46.         {  
  47.             //xml元素节点开始时触发,是“person”  
  48.             person = new Person();  
  49.             person.setId(new Integer(attributes.getValue(0)));  
  50.         }  
  51.         tag =localName;//保存元素节点名称  
  52.     }  
  53.     @Override  
  54.     public void endElement(String uri, String localName, String qName)  
  55.             throws SAXException {  
  56.         //元素节点结束时触发,是“person”  
  57.         if("person".equals(localName))  
  58.         {  
  59.             list.add(person);  
  60.             person=null;  
  61.         }  
  62.         tag =null;//结束时,需要清空tag  
  63.         }  
  64.     @Override  
  65.     public void characters(char[] ch, int start, int length)  
  66.             throws SAXException {  
  67.         if(tag!=null)  
  68.         {  
  69.             String data = new String(ch,start,length);  
  70.            if("name".equals(tag))  
  71.            {  
  72.                person.setName(data);  
  73.                  
  74.            }else if("age".equals(tag))  
  75.            {  
  76.                person.setAge(new Integer(data));  
  77.            }  
  78.         }  
  79.     }  
  80.   
  81.       
  82.   
  83.       
  84.          
  85.          
  86.    }  
  87. }  

3)

 Pull解析和Sax解析很相似,都是轻量级的解析,在Android的内核中已经嵌入了Pull,所以我们不需要再添加第三方jar包来支持Pull。Pull解析和Sax解析不一样的地方有(1)pull读取xml文件后触发相应的事件调用方法返回的是数字(2)pull可以在程序中控制想解析到哪里就可以停止解析。

[java]  view plain copy
  1. package cn.itcast.service;  
  2.   
  3. import java.io.InputStream;  
  4. import java.io.Writer;  
  5. import java.util.ArrayList;  
  6. import java.util.List;  
  7.   
  8. import org.xmlpull.v1.XmlPullParser;  
  9. import org.xmlpull.v1.XmlSerializer;  
  10.   
  11. import android.util.Xml;  
  12.   
  13. import cn.itcast.model.Person;  
  14.   
  15. public class PullPersonService {  
  16.     //保存xml文件  
  17.     public static void saveXML(List<Person> list,Writer write)throws Throwable  
  18.     {  
  19.         XmlSerializer serializer =Xml.newSerializer();//序列化  
  20.         serializer.setOutput(write);//输出流  
  21.         serializer.startDocument("UTF-8"true);//开始文档  
  22.         serializer.startTag(null"persons");  
  23.         //循环去添加person  
  24.         for (Person person : list) {  
  25.             serializer.startTag(null"person");  
  26.             serializer.attribute(null"id", person.getId().toString());//设置id属性及属性值  
  27.             serializer.startTag(null"name");  
  28.             serializer.text(person.getName());//文本节点的文本值--name  
  29.             serializer.endTag(null"name");  
  30.             serializer.startTag(null"age");  
  31.             serializer.text(person.getAge().toString());//文本节点的文本值--age  
  32.             serializer.endTag(null"age");  
  33.             serializer.endTag(null"person");  
  34.         }  
  35.         serializer.endTag(null"persons");  
  36.         serializer.endDocument();  
  37.         write.flush();  
  38.         write.close();  
  39.     }  
  40.      public List<Person> getPersons(InputStream stream) throws Throwable  
  41.      {  
  42.          List<Person> list =null;  
  43.         Person person =null;  
  44.         XmlPullParser parser =Xml.newPullParser();  
  45.         parser.setInput(stream,"UTF-8");  
  46.         int type =parser.getEventType();//产生第一个事件  
  47.         //只要当前事件类型不是”结束文档“,就去循环  
  48.         while(type!=XmlPullParser.END_DOCUMENT)  
  49.         {  
  50.         switch (type) {  
  51.         case XmlPullParser.START_DOCUMENT:  
  52.         list =    new ArrayList<Person>();  
  53.             break;  
  54.   
  55.         case XmlPullParser.START_TAG:  
  56.             String name=parser.getName();//获取解析器当前指向的元素名称  
  57.             if("person".equals(name))  
  58.             {  
  59.                 person =new Person();  
  60.                 person.setId(new Integer(parser.getAttributeValue(0)));  
  61.             }  
  62.             if(person!=null)  
  63.             {  
  64.                 if("name".equals(name))  
  65.                 {  
  66.                     person.setName(parser.nextText());//获取解析器当前指向的元素的下一个文本节点的文本值  
  67.                 }  
  68.                 if("age".equals(name))  
  69.                 {  
  70.                     person.setAge(new Integer(parser.nextText()));  
  71.                 }  
  72.             }  
  73.             break;  
  74.         case XmlPullParser.END_TAG:  
  75.             if("person".equals(parser.getName()))  
  76.             {  
  77.                 list.add(person);  
  78.                 person=null;  
  79.             }  
  80.             break;  
  81.         }  
  82.         type=parser.next();//这句千万别忘了哦  
  83.         }  
  84.          return list;  
  85.      }  
  86. }  

下面是Model层的Person类的代码:

[java]  view plain copy
  1. package cn.itcast.model;  
  2.   
  3.   
  4.   
  5. public class Person {  
  6. private Integer id;  
  7. public Integer getId() {  
  8.     return id;  
  9. }  
  10.   
  11.   
  12. public void setId(Integer id) {  
  13.     this.id = id;  
  14. }  
  15.   
  16. private String name;  
  17. public String getName() {  
  18.     return name;  
  19. }  
  20.   
  21.   
  22. public void setName(String name) {  
  23.     this.name = name;  
  24. }  
  25.   
  26. private Integer age;  
  27. public Integer getAge() {  
  28.     return age;  
  29. }  
  30.   
  31.   
  32. public void setAge(Integer age) {  
  33.     this.age = age;  
  34. }  
  35.   
  36.   
  37.   
  38.   
  39.   
  40. public Person()  
  41. {  
  42. }  
  43. public Person(Integer id, String name, Integer age) {  
  44.       
  45.     this.id = id;  
  46.     this.name = name;  
  47.     this.age = age;  
  48. }  
  49.   
  50.   
  51. @Override  
  52. public String toString() {  
  53.     return "Person [id=" + id + ", name=" + name + ", age=" + age + "]";  
  54. }  
  55.   
  56. }  
xml文件操作 public class XmlUtils { /** * 获取Document对象。根据xml文件的名字获取Document对象。 * * @param file * 要获取对象的xml文件全路径。 * @return 返回获取到的Document对象。 * @throws IOException * 如果发生任何 IO 错误抛出此异常。 * @throws SAXException * 如果发生任何解析错误抛出此异常。 * @throws ParserConfigurationException * 如果无法创建满足所请求配置的 DocumentBuilder,将抛出该异常。 * @exception NullPointerException * 如果file为空,抛出此异常。 */ public static Document parseForDoc(final String file) throws SAXException, IOException, SecurityException, NullPointerException, ParserConfigurationException { return XmlUtils.parseForDoc(new FileInputStream(file)); } /** * 将一个xml字符串解析成Document对象。 * * @param xmlStr * 要被解析xml字符串。 * @param encoding * 字符串的编码。 * @return 返回解析后的Document对象。 * @throws IOException * 如果发生任何 IO 错误抛出此异常。 * @throws SAXException * 如果发生任何解析错误抛出此异常。 * @throws ParserConfigurationException * 如果无法创建满足所请求配置的 DocumentBuilder,将抛出该异常。 */ public static Document parseForDoc(String xmlStr, String encoding) throws SAXException, IOException, ParserConfigurationException { if (xmlStr == null) { xmlStr = ""; } ByteArrayInputStream byteInputStream = new ByteArrayInputStream( xmlStr.getBytes(encoding)); return XmlUtils.parseForDoc(byteInputStream); } /** * 获取Document对象。根据字节输入流获取一个Document对象。 * * @param is * 获取对象的字节输入流。 * @return 返回获取到的Document对象。如果出现异常,返回null。 * @throws IOException * 如果发生任何 IO 错误抛出此异常。 * @throws SAXException * 如果发生任何解析错误抛出此异常。 * @throws ParserConfigurationException * 如果无法创建满足所请求配置的 DocumentBuilder,将抛出该异常。 * @exception IllegalArgumentException * 当 is 为 null 抛出此异常。 */ public static Document parseForDoc(final InputStream is) throws SAXException, IOException, ParserConfigurationException, IllegalArgumentException { try { DocumentBuilderFactory factory = DocumentBuilderFactory .newInstance(); DocumentBuilder builder = factory.newDocumentBuilder(); return builder.parse(is); } finally { is.close(); } } /** * 通过xpath表达式解析某个xml节点。 * * @param obj * 要被解析xml节点对象。 * @param xPath * xpath表达式。 * @param qName * 被解析的目标类型。 * @return 返回解析后的对象。 * @throws XPathExpressionException * 如果不能计算 expression。 * * @exception RuntimeException * 创建默认对象模型的 XPathFactory 遇到故障。 * @exception NullPointerException * 如果xPath为空抛出异常。 */ private static Object parseByXpath(final Object obj, final String xPath, QName qName) throws NullPointerException, RuntimeException, XPathExpressionException { XPathFactory xpathFactory = XPathFactory.newInstance(); XPath path = xpathFactory.newXPath(); return path.evaluate(xPath, obj, qName); } /** * 通过XPath表达式获取单个节点。 * * @param obj * 要被解析的對象。 * @param xPath * XPath表达式。 * @return 返回获取到的节点。 * * @throws XPathExpressionException * 如果不能计算 expression。 * * @exception RuntimeException * 创建默认对象模型的 XPathFactory 遇到故障。 * @exception NullPointerException * 如果xPath为空抛出异常。 */ public static Node parseForNode(final Object obj, final String xPath) throws NullPointerException, RuntimeException, XPathExpressionException { return (Node) XmlUtils.parseByXpath(obj, xPath, XPathConstants.NODE); } /** * 通过XPath表达式获取某个xml节点的字符串值。 * * @param obj * 要被解析的對象。 * @param xPath * XPath表达式。 * @return 返回获取到的节点的字符串值。 * * @throws XPathExpressionException * 如果不能计算 expression。 * * @exception RuntimeException * 创建默认对象模型的 XPathFactory 遇到故障。 * @exception NullPointerException * 如果xPath为空抛出异常。 */ public static String parseForString(final Object obj, final String xPath) throws NullPointerException, RuntimeException, XPathExpressionException { return (String) XmlUtils .parseByXpath(obj, xPath, XPathConstants.STRING); } /** * 通过XPath表达式获取某个xml节点的布尔值。 * * @param obj * 要被解析的對象。 * @param xPath * XPath表达式。 * @return 返回获取到的节点的布尔值。 * * @throws XPathExpressionException * 如果不能计算 expression。 * * @exception RuntimeException * 创建默认对象模型的 XPathFactory 遇到故障。 * @exception NullPointerException * 如果xPath为空抛出异常。 */ public static Boolean parseForBoolean(final Object obj, final String xPath) throws NullPointerException, RuntimeException, XPathExpressionException { return (Boolean) XmlUtils.parseByXpath(obj, xPath, XPathConstants.BOOLEAN); } /** * 通过XPath表达式获取Node列表。 * * @param obj * 要被解析的對象。 * @param xPath * XPath表达式。 * @return 返回获取到的Node列表。 * * @throws XPathExpressionException * 如果不能计算 expression。 * * @exception RuntimeException * 创建默认对象模型的 XPathFactory 遇到故障。 * @exception NullPointerException * 如果xPath为空抛出异常。 */ public static List parseForNodeList(final Object obj, final String xPath) throws NullPointerException, RuntimeException, XPathExpressionException { List lists = new ArrayList(); NodeList nList = (NodeList) XmlUtils.parseByXpath(obj, xPath, XPathConstants.NODESET); if (nList != null) { for (int i = 0; i < nList.getLength(); i++) { lists.add(nList.item(i)); } } return lists; } /** * 获取节点的制定属性。 * * @param node * 节点。 * @param attrName * 属性名。 * @return 返回获取到的属性值。如果找不到相关的 * */ public static String getAttribute(final Object node, final String attrName) { String result = ""; if ((node != null) && (node instanceof Node)) { if (((Node) node).getNodeType() == Node.ELEMENT_NODE) { result = ((Element) node).getAttribute(attrName); } else { // 遍历整个xml某节点指定的属性 NamedNodeMap attrs = ((Node) node).getAttributes(); if ((attrs.getLength() > 0) && (attrs != null)) { Node attr = attrs.getNamedItem(attrName); result = attr.getNodeValue(); } } } return result; } /** * 使用新节点替换原来的旧节点。 * * @param oldNode * 要被替换的旧节点。 * @param newNode * * 替换后的新节点。 * @exception DOMException * 如果此节点为不允许 * newNode节点类型的子节点的类型;或者如果要放入的节点为此节点的一个祖先或此节点本身;或者如果此节点为 * Document 类型且替换操作的结果将第二个 DocumentType 或 Element 添加到 * Document 上。 WRONG_DOCUMENT_ERR: 如果 newChild * 是从不同的文档创建的,不是从创建此节点的文档创建的,则引发此异常。 * NO_MODIFICATION_ALLOWED_ERR: 如果此节点或新节点的父节点为只读的,则引发此异常。 * NOT_FOUND_ERR: 如果 oldChild 不是此节点的子节点,则引发此异常。 * NOT_SUPPORTED_ERR: 如果此节点为 Document 类型,则如果 DOM 实现不支持替换 * DocumentType 子节点或 Element 子节点,则可能引发此异常。 */ public static void replaceNode(Node oldNode, Node newNode) { if ((oldNode != null) && (newNode != null)) { oldNode.getParentNode().replaceChild(newNode, oldNode); } } /** * 将Document输出到指定的文件。 * * @param fileName * 文件名。 * @param node * 要保存的对象。 * @param encoding * 保存的编码。 * @throws FileNotFoundException * 指定的文件名不存在,抛出此异常。 * @throws TransformerException * 如果转换过程发生不可恢复的错误,抛出此异常。 */ public static void saveXml(final String fileName, final Node node, String encoding) throws FileNotFoundException, TransformerException { XmlUtils.writeXml(new FileOutputStream(fileName), node, encoding); } /** * 将Document输出成字符串的形式。 * * @param node * Node对象。 * @param encoding * 字符串的编码。 * @return 返回输出成的字符串。 * @throws TransformerException * 如果转换过程发生不可恢复的错误,抛出此异常。 * @throws UnsupportedEncodingException * 指定的字符串编码不支持,抛出此异常。 */ public static String nodeToString(Node node, String encoding) throws TransformerException, UnsupportedEncodingException { ByteArrayOutputStream outputStream = new ByteArrayOutputStream(); XmlUtils.writeXml(outputStream, node, encoding); return outputStream.toString(encoding); } /** * 将指定的Node写到指定的OutputStream流。 * * @param encoding * 编码。 * @param os * OutputStream流。 * @param node * Node节点。 * @throws TransformerException * 如果转换过程发生不可恢复的错误,抛出此异常。 */ private static void writeXml(OutputStream os, Node node, String encoding) throws TransformerException { TransformerFactory transFactory = TransformerFactory.newInstance(); Transformer transformer = transFactory.newTransformer(); transformer.setOutputProperty("indent", "yes"); transformer.setOutputProperty(OutputKeys.ENCODING, encoding); DOMSource source = new DOMSource(); source.setNode(node); StreamResult result = new StreamResult(); result.setOutputStream(os); transformer.transform(source, result); } }
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值