XML解析

一、概念

XML,英文全称为Extensible Markup Language,可扩展标记语言。XML只定义了文档的结构,如成对的标记、属性的位置等,至于这些标记和属性解析成什么由开发者自己决定。在Android中,XML解析有三种方式,DOM解析、SAX解析和PULL解析。

XML文件例子:(位于项目assets路径下,文件名为example.xml)


    
        1
        Google Maps
        1.0
    
    
        2
        Chrome
        2.1
    
    
        3
        Google Play
        2.3
    

二、DOM解析

优点:
● 形成了树结构,有助于更好的理解、掌握,且代码容易编写。
● 解析过程中,树结构保存在内存中,方便修改。

缺点:
● 由于文件是一次性读取,所以对内存的耗费比较大。
● 如果XML文件比较大,容易影响解析性能且可能会造成内存溢出。

例子:

private void xml_dom(){
    StringBuilder data = new StringBuilder();
    try {
        //打开assets文件夹下的xml示例文件到输入流
        InputStream in = getAssets().open("example.xml");
        //得到Document Builder Factory对象
        DocumentBuilderFactory builderFactory = DocumentBuilderFactory.newInstance();
        //得到Document Builder对象
        DocumentBuilder builder = builderFactory.newDocumentBuilder();
        //得到Document存放整个xml的Document对象数据
        Document document = builder.parse(in);
        //得到xml数据的根节点
        Element rootElement = document.getDocumentElement();
        //得到根节点下所有app节点
        NodeList list = rootElement.getElementsByTagName("app");
        //遍历所有节点
        for (int i = 0;i < list.getLength(); i++){
            //获取app节点的所有子元素
            Element app = (Element) list.item(i);
            //获取app节点的子元素id,name,version,并添加到StringBuilder中
            data.append("id is " + app.getElementsByTagName("id").item(0).getTextContent() + "\n");
            data.append("name is " + app.getElementsByTagName("name").item(0).getTextContent() + "\n");
            data.append("version is " + app.getElementsByTagName("version").item(0).getTextContent() + "\n");
        }
        //更新UI
        dataText.setText(data);
    }catch (Exception e){
        e.printStackTrace();
    }
}

三、SAX解析

优点:
● 采用事件驱动模式,解析速度快,占用内存少。
● 非常适合在Android移动设备中使用。

缺点:
● 编码比较麻烦。
● 很难同时访问XML文件中的多处不同数据。

例子:

class ContentHandler extends DefaultHandler{
    private String nodeName;
    private StringBuilder id;
    private StringBuilder name;
    private StringBuilder version;
    private StringBuilder text;

    //在开始XML解析的时候调用
    @Override
    public void startDocument() throws SAXException {
        id = new StringBuilder();
        name = new StringBuilder();
        version = new StringBuilder();
        text = new StringBuilder();
    }

    //在解析某个节点的时候调用
    @Override
    public void startElement(String uri, String localName, String qName, Attributes attributes) throws SAXException {
        nodeName = localName;
    }

    //在获取节点内容时调用
    //注意:获取内容时,该方法可能会被调用多次,同时换行符也会被当作内容解析出来
    @Override
    public void characters(char[] ch, int start, int length) throws SAXException {
        if ("id".equals(nodeName)){
            id.append(ch, start, length);
        }else if ("name".equals(nodeName)){
            name.append(ch, start, length);
        }else if ("version".equals(nodeName)){
            version.append(ch, start, length);
        }
    }

    //在对某个节点的解析完成时调用
    @Override
    public void endElement(String uri, String localName, String qName) throws SAXException {
        if ("app".equals(localName)){
            text.append("id is " + id.toString() + "\n");
            text.append("name" + name.toString() + "\n");
            text.append("version is " + version.toString() + "\n");
            id.setLength(0);
            name.setLength(0);
            version.setLength(0);
        }
    }

    //整个XML解析完成的时候调用
    @Override
    public void endDocument() throws SAXException {
        dataText.setText(text);
        super.endDocument();

    }
}

private void xml_sax(){
    try {
        SAXParserFactory factory = SAXParserFactory.newInstance();
        SAXParser parser = factory.newSAXParser();
        ContentHandler mHandler = new ContentHandler();
        //打开assets文件夹下的xml示例文件到输入流
        InputStream in = getAssets().open("example.xml");
        parser.parse(in,mHandler);

    } catch (Exception e) {
        e.printStackTrace();

    }
}

四、PULL解析

优点:
● PULL解析器小巧轻便,解析速度快,简单易用。
● 灵活性高,自由控制访问时机。
● 非常适合在Android移动设备中使用,Android系统内部在解析各种XML时也是用PULL解析器。

缺点:
● 可扩展性差,无法修改XML树内容结构。

例子:

private void xml_pull(){
    /*
    try {
        XmlPullParserFactory factory = XmlPullParserFactory.newInstance();
        XmlPullParser parser = factory.newPullParser();
        parser.setInput(new StringReader(xmlData));
    } catch (XmlPullParserException e) {
        e.printStackTrace();
    }*/

    StringBuilder builder = new StringBuilder("");
    XmlPullParser parser = Xml.newPullParser();

    try {
        //打开assets文件夹下的xml示例文件到输入流
        InputStream in = getAssets().open("example.xml");
        parser.setInput(in,"UTF-8");
        while (parser.getEventType() != XmlPullParser.END_DOCUMENT){
            String nodeName = parser.getName();
            int eventType = parser.getEventType();
            switch (eventType){
                case XmlPullParser.START_DOCUMENT:
                    break;
                case XmlPullParser.START_TAG:
                    if ("id".equals(nodeName)){
                        builder.append("id is : " + parser.nextText() + "\n");
                    }else if ("name".equals(nodeName)){
                        builder.append("name is : "+ parser.nextText() + "\n");
                    }else if ("version".equals(nodeName)){
                        builder.append("version is : "+ parser.nextText() + "\n");
                    }
                    break;
                case XmlPullParser.END_TAG:
                    break;
            }
            parser.next();
        }
        dataText.setText(builder.toString());
    } catch (Exception e) {
        e.printStackTrace();
    }
} 

你可能感兴趣的:(XML解析)