新手使用android的XML(DOM)解析问题,指定路径XML如何读取

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了新手使用android的XML(DOM)解析问题,指定路径XML如何读取相关的知识,希望对你有一定的参考价值。

Log.i("通知:", "OpenXml");
DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance();
DocumentBuilder builder = factory.newDocumentBuilder();
Log.i("通知:", "getAssetsopen");
InputStream inputStream = this.getResources().getAssets().open("/sdcard/TB/index.xml");
Document document = builder.parse(inputStream);

如上图所示我在this.getResources().getAssets().open这句通不过,我也知道这个open好像是为指定APK内资源文件中包含的XML,如果我希望读取一个SD卡上的XML我应该如何做。求高手指点

一、在android应用中的XML文件来源
1、本地xml文件
本地XML文件可以放在应用根目录assets文件夹、res/xml、res/raw、SDcard卡、应用的data目录等;
除res/xml可直接通过getXml(int id)获取XML文档,返回一个解析器对象(XmlResourceParer:XmlResourceParer是XmlPullParser的子类),其它位置情况都可以获取XML文档,返回一个Inputstream对象,进行读取数据,获取方法分别如下:
a.在res/xml目录下(推荐使用):
[java] view plaincopy
XmlResourceParser xmlParser = this.getResources().getXml(R.xml.XXX);

b.在res/xml、res/raw目录下:
[java] view plaincopy
InputStream inputStream = this.getResources().openRawResource(R.xml.XXX);

c.在assets文件夹下(本人测试发现通过此方法获取的XML文档不能带有首行:<?xml version="1.0" encoding="utf-8"?>,否则解析报错,具体原因未查明,知道原因请回复交流):
[java] view plaincopy
InputStream inputStream = getResources().getAssets().open(fileName);

d.在应用指定目录下(SDcard,应用data目录等):
[java] view plaincopy
// path路径根据实际项目修改,此次获取SDcard根目录
String path = Environment.getExternalStorageDirectory().toString();
File xmlFlie = new File(path+fileName);
InputStream inputStream = new FileInputStream(xmlFlie);

2、通过url得到的xml文件
很多时候需要解析xml文件都用于客户端与服务器之间的数据交互,比如解析google天气预报信息,或自己项目内定的一些XML数据结构,其中通过URL,使用DefaultHTTPClient get请求获取XML文件方法如下:
[java] view plaincopy
/**
* 读取url的xml资源 转成String
* @param url
* @return 返回 读取url的xml字符串
*/
public String getStringByUrl(String url)
String outputString = "";
// DefaultHttpClient
DefaultHttpClient httpclient = new DefaultHttpClient();
// HttpGet
HttpGet httpget = new HttpGet(url);
// ResponseHandler
ResponseHandler<String> responseHandler = new BasicResponseHandler();

try
outputString = httpclient.execute(httpget, responseHandler);
outputString = new String(outputString.getBytes("ISO-8859-1"), "utf-8"); // 解决中文乱码

Log.i("HttpClientConnector", "连接成功");
catch (Exception e)
Log.i("HttpClientConnector", "连接失败");
e.printStackTrace();

httpclient.getConnectionManager().shutdown();
return outputString;


二、XML文件的解析方式
能够运用在Android系统上解析XML文件的常用有三种方式:DOM、SAX和PULL,其中DOM解析XML是先把XML文件读进内存中,再通过接口获取数据,该方法使用相对小的XML文件,移动设备往往受硬件性能影响,如果XML文件比较大使用DOM解析往往效率跟不上;SAX和PULL都是采用事件驱动方式来进行解析,在Android中的事件机制是基于回调函数。
本例旨在考虑简单方便性,综合考虑选择了PULL解析,PULL解析器是一个开源项目,Android平台已经内置了PULL解析器,同时Android系统本身也是使用PULL解析器来解析各种XML文档。
1、事件回调类型
PULL解析XML文件时,回调XmlResourceParser内定义表示文档开头结束和节点开头结束的数值(事件回调类型),表示如下:
a.读取到XML文档开头(声明)返回:XmlPullParser.START_DOCUMENT(0)
b.读取到XML文档结束返回:XmlPullParser.END_DOCUMENT (1)
c.读取到XML节点开始返回:XmlPullParser.START_TAG (2)
d.读取到XML节点结束返回:XmlPullParser.END_TAG (3)
e.读取到XML文本返回:XmlPullParser.TEXT (4)

2、XmlPullParser有几个主要方法(更多查阅Android APIs):
a.XmlPullParser.getEventType() : Returns the type of the current event (START_TAG, END_TAG, TEXT, etc.) 【获取当前事件回调类型】
b.XmlPullParser.getName():For START_TAG or END_TAG events, the (local) name of the current element is returned when namespaces are enabled.【获取当前节点名字】
c.XmlPullParser.getAttributeValue(int index):Returns the given attributes value.【根据id获取节点属性值】
d.XmlPullParser.getAttributeValue(String namespace, String name):Returns the attributes value identified by namespace URI and namespace localName.【根据name获取节点属性值】
e.XmlPullParser.netxText():If current event is START_TAG then if next element is TEXT then element content is returned or if next event is END_TAG then empty string is returned, otherwise exception is thrown.【回调节点START_TAG时,通过此方法获取节点内容】
3、实际编码中如何使用
在实际编码中,主要根据事件回调类型,结合被解析的XML结构进行解析提取数据,PULL解析XML文件的主要模式如下,更具体使用看本文提供的例子:
[java] view plaincopy
try
//开始解析事件
int eventType = parser.getEventType();

//处理事件,不碰到文档结束就一直处理
while (eventType != XmlPullParser.END_DOCUMENT)
//因为定义了一堆静态常量,所以这里可以用switch
switch (eventType)
case XmlPullParser.START_DOCUMENT:
// 不做任何操作或初开始化数据
break;

case XmlPullParser.START_TAG:
// 解析XML节点数据
// 获取当前标签名字
String tagName = parser.getName();

if(tagName.equals("XXXTAGXXX"))

// 通过getAttributeValue 和 netxText解析节点的属性值和节点值


break;

case XmlPullParser.END_TAG:
// 单节点完成,可往集合里边添加新的数据
break;
case XmlPullParser.END_DOCUMENT:

break;


// 别忘了用next方法处理下一个事件,不然就会死循环
eventType = parser.next();

catch (XmlPullParserException e)
e.printStackTrace();
catch (IOException e)
e.printStackTrace();
参考技术A File file = new File(Environment.getExternalStorageDirectory(),“文件地址”);
InputStream is = new FileInputStream(file);
Document doc = builder.parse(is);
还有最好判断一下sd卡状态
if (Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED))
在进行SD卡操作。本回答被提问者采纳

如何在 android 中使用 DOM 或 SAX 解析器从 XML 读取子节点

【中文标题】如何在 android 中使用 DOM 或 SAX 解析器从 XML 读取子节点【英文标题】:How to read the child nodes from XML using DOM or SAX parser in android 【发布时间】:2013-05-18 08:39:24 【问题描述】:

这是我的 XML。

<Operations>
<Operation Name="OperationName1">Entity details1</Operation>
<Operation Name="OperationName2">Entity details2</Operation>
<Operation Name="OperationName3">Entity details3</Operation>
<Operation Name="OperationName4">Entity details4</Operation>
</Operations>

在此我需要将每个子节点作为字符串变量读取。使用 DOM 我正在​​尝试这样。

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

        for (int i=0;i<items.getLength();i++)
           
            Node item = items.item(i);

            NodeList properties = item.getChildNodes();

            for (int j=0;j<properties.getLength();j++)

                Node property = properties.item(j);

                                   
         

现在据我所知,这些项目现在拥有所有子节点,我需要像这样存储每个子节点。

String ch_node="<Operation Name="OperationName4">Entity details4</Operation>"

是否有任何默认方法可以为我提供子节点 xml,或者我需要使用节点名称、值和属性再次创建?

我也尝试过使用 SAX 解析器,但不知道如何获取。

public void startElement(String uri, String localName, String qName,
        Attributes attributes) throws SAXException        
    if (qName.equalsIgnoreCase("operation"))      
        op_Name=attributes.getValue(0);
    


public void characters(char[] ch, int start, int length)
        throws SAXException         


public void endElement(String uri, String localName, String qName)
        throws SAXException        

【问题讨论】:

【参考方案1】:

你可以试试 DOM 和 Transformer

    Transformer tx = TransformerFactory.newInstance().newTransformer();
    tx.setOutputProperty(OutputKeys.OMIT_XML_DECLARATION, "yes");
    Document doc = DocumentBuilderFactory.newInstance().newDocumentBuilder().parse(new File("1.xml"));
    NodeList list = doc.getElementsByTagName("Operation");
    for (int i = 0; i < list.getLength(); i++) 
        DOMSource src = new DOMSource(list.item(i));
        StringWriter sr = new StringWriter();
        Result res = new StreamResult(sr);
        tx.transform(src, res);
        System.out.println(sr);
    

输出

<Operation Name="OperationName1">Entity details1</Operation>
<Operation Name="OperationName2">Entity details2</Operation>
<Operation Name="OperationName3">Entity details3</Operation>
<Operation Name="OperationName4">Entity details4</Operation>

【讨论】:

没有任何默认方法可以给子节点xml吗?因为如果我有非常大的 xml 并且每个子节点都有不同的节点名称和属性,那么这将变得更加复杂。 好的。看看新版本【参考方案2】:

试试这个

String elemName;

public void startElement(String uri, String localName, String qName,
        Attributes attributes) throws SAXException        
   elemName=qName;


public void characters(char[] ch, int start, int length)
        throws SAXException   
if(elemName.equals("OperationName1")) 
 String OperationName1Text=new String(ch);
     


public void endElement(String uri, String localName, String qName)
        throws SAXException        

【讨论】:

抱歉,但这只会返回“Entity Details2”之类的文本,还是我错了? @Giri 是字符方法只返回它。其他方法如开始元素将给出元素名称及其属性。你还想得到什么 谢谢,就像我在帖子中解释的那样,我需要将子节点的整个 xml 存储在字符串变量中。我们可以像 Evgeniy Dorofeev 所说的那样创建它。但我有非常复杂的 xml。所以我正在尝试使用一些默认方法。有可能吗?【参考方案3】:

请看下面的代码,这里我获取了一个子节点“描述”

URL url;

try 

    url = new URL(urls);
    HttpURLConnection conn = (HttpURLConnection) url.openConnection();
    if ((conn.getResponseCode() == HttpURLConnection.HTTP_OK)) 
        DocumentBuilderFactory dbf = DocumentBuilderFactory
                .newInstance();
        DocumentBuilder db = dbf.newDocumentBuilder();
        Document doc;
        doc = db.parse(url.openStream());
        doc.getDocumentElement().normalize();

        NodeList itemLst = doc.getElementsByTagName("item");
        nl = doc.getElementsByTagName(KEY_HEAD);

        Description = new String[itemLst.getLength()];// ........


        for (int i = 0; i < itemLst.getLength(); i++) 

            Node item = itemLst.item(i);
            if (item.getNodeType() == Node.ELEMENT_NODE) 
                Element ielem = (Element) item;

                NodeList description = ielem
                        .getElementsByTagName("description");

                Desc[i] = description.item(0).getChildNodes().item(0)
                        .getNodeValue();

            

        

    
 catch (MalformedURLException e) 
    // TODO Auto-generated catch block
    e.printStackTrace();
 catch (DOMException e) 
    // TODO Auto-generated catch block
    e.printStackTrace();
 catch (IOException e) 
    // TODO Auto-generated catch block
    e.printStackTrace();
 catch (ParserConfigurationException e) 
    // TODO Auto-generated catch block
    e.printStackTrace();
 catch (SAXException e) 
    // TODO Auto-generated catch block
    e.printStackTrace();

【讨论】:

对不起,但这也只返回节点内的文本。

以上是关于新手使用android的XML(DOM)解析问题,指定路径XML如何读取的主要内容,如果未能解决你的问题,请参考以下文章

Android之DOM解析XML

Android之DOM解析XML

Android -使用DOM(文档对象模型)解析XML文件

Android Dom和XmlPullParse解析XML文件

用于 XML 编辑、更新的 DOM 解析器

如何在 android 中使用 DOM 或 SAX 解析器从 XML 读取子节点