android pull更新xml数据,Android 使用Pull方法解析XML文件的方法

Pull解析方法给应用程序完全的控制文档该怎么样被解析。Android中对Pull方法提供了支持的API,主要是

org.xmlpull.v1.XmlPullParser;

org.xmlpull.v1.XmlPullParserFactory;

二个类,其中主要使用的是XmlPullParser,XmlPullParserFactory是一个工厂,用于构建XmlPullParser对象。

应用程序通过调用XmlPullParser.next()等方法来产生Event,然后再处理Event。可以看到它与Push方法的不同,Push方法是由Parser自己主动产生Event,回调给应用程序。而Pull方法是主动的调用Parser的方法才能产生事件。

假如XML中的语句是这样的:"James Elliott",author是TAG,country是ATTRIBUTE,"James Elliott"是TEXT。

要想解析文档先要构建一个XmlPullParser对象

final XmlPullParserFactory factory = XmlPullParserFactory.newInstance();

factory.setNamespaceAware(true);

final XmlPullParser parser = factory.newPullParser();

Pull解析是一个遍历文档的过程,每次调用next(),nextTag(), nextToken()和nextText()都会向前推进文档,并使Parser停留在某些事件上面,但是不能倒退。

然后把文档设置给Parser

parser.setInput(new StringReader("James Elliott");

这时,文档刚被初始化,所以它应该位于文档的开始,事件应该是START_DOCUMENT,可以通过XmlPullParser.getEventType()来获取。然后调用next()会产生

START_TAG,这个事件告诉应用程序一个标签已经开始了,调用getName()会返回"author";再next()会产生

TEXT事件,调用getText()会返回"James Elliott",再next(),会产生

END_TAG,这个告诉你一个标签已经处理完了,再next(),会产生

END_DOCUMENT,它告诉你整个文档已经处理完成了。

除了next()外,nextToken()也可以使用,只不过它会返回更加详细的事件,比如 COMMENT, CDSECT, DOCDECL, ENTITY等等非常详细的信息。如果程序得到比较底层的信息,可以用nextToken()来驱动并处理详细的事件。需要注意一点的是TEXT事件是有可能返回空白的White Spaces比如换行符或空格等。

另外有二个非常实用的方法nextTag()和nextText()

nextTag()--首先它会忽略White Spaces,如果可以确定下一个是START_TAG或END_TAG,就可以调用nextTag()直接跳过去。通常它有二个用处:当START_TAG时,如果能确定这个TAG含有子TAG,那么就可以调用nextTag()产生子标签的START_TAG事件;当END_TAG时,如果确定不是文档结尾,就可以调用nextTag()产生下一个标签的START_TAG。在这二种情况下如果用next()会有TEXT事件,但返回的是换行符或空白符。

nextText()--它只能在START_TAG时调用。当下一个元素是TEXT时,TEXT的内容会返回;当下一个元素是END_TAG时,也就是说这个标签的内容为空,那么空字串返回;这个方法返回后,Parser会停在END_TAG上。比如:

James Elliott

当START_TAG时,调用nextText(),依次返回:

"James Elliott"

""(empty)

""(empty)这个方法在处理没有子标签的标签时很有用。比如:

What Is Hibernate

James Elliott

Web

就可以用以下代码来处理:

while (eventType != XmlPullParser.END_TAG) {

switch (eventType) {

case XmlPullParser.START_TAG:

tag = parser.getName();

final String content = parser.nextText();

Log.e(TAG, tag + ": [" + content + "]");

eventType = parser.nextTag();

break;

default:

break;

}

}

这就要比用next()来处理方便多了,可读性也大大的加强了。

最后附上一个解析XML的实例Android程序

import java.io.IOException;

import java.io.InputStream;

import org.xmlpull.v1.XmlPullParser;

import org.xmlpull.v1.XmlPullParserException;

import org.xmlpull.v1.XmlPullParserFactory;

import android.util.Log;

public class RssPullParser extends RssParser {

private final String TAG = FeedSettings.GLOBAL_TAG;

private InputStream mInputStream;

public RssPullParser(InputStream is) {

mInputStream = is;

}

public void parse() throws ReaderBaseException, XmlPullParserException, IOException {

if (mInputStream == null) {

throw new ReaderBaseException("no input source, did you initialize this class correctly?");

}

final XmlPullParserFactory factory = XmlPullParserFactory.newInstance();

factory.setNamespaceAware(true);

final XmlPullParser parser = factory.newPullParser();

parser.setInput(mInputStream);

int eventType = parser.getEventType();

if (eventType != XmlPullParser.START_DOCUMENT) {

throw new ReaderBaseException("Not starting with 'start_document'");

}

eventType = parseRss(parser);

if (eventType != XmlPullParser.END_DOCUMENT) {

throw new ReaderBaseException("not ending with 'end_document', do you finish parsing?");

}

if (mInputStream != null) {

mInputStream.close();

} else {

Log.e(TAG, "inputstream is null, XmlPullParser closed it??");

}

}

/**

* Parsing the Xml document. Current type must be Start_Document.

* After calling this, Parser is positioned at END_DOCUMENT.

* @param parser

* @return event end_document

* @throws XmlPullParserException

* @throws ReaderBaseException

* @throws IOException

*/

private int parseRss(XmlPullParser parser) throws XmlPullParserException, ReaderBaseException, IOException {

int eventType = parser.getEventType();

if (eventType != XmlPullParser.START_DOCUMENT) {

throw new ReaderBaseException("not starting with 'start_document', is this a new document?");

}

Log.e(TAG, "starting document, are you aware of that!");

eventType = parser.next();

while (eventType != XmlPullParser.END_DOCUMENT) {

switch (eventType) {

case XmlPullParser.START_TAG: {

Log.e(TAG, "start tag: '" + parser.getName() + "'");

final String tagName = parser.getName();

if (tagName.equals(RssFeed.TAG_RSS)) {

Log.e(TAG, "starting an RSS feed <

final int attrSize = parser.getAttributeCount();

for (int i = 0; i < attrSize; i++) {

Log.e(TAG, "attr '" + parser.getAttributeName(i) + "=" + parser.getAttributeValue(i) + "'");

}

} else if (tagName.equals(RssFeed.TAG_CHANNEL)) {

Log.e(TAG, "\tstarting an Channel <

parseChannel(parser);

}

break;

}

case XmlPullParser.END_TAG: {

Log.e(TAG, "end tag: '" + parser.getName() + "'");

final String tagName = parser.getName();

if (tagName.equals(RssFeed.TAG_RSS)) {

Log.e(TAG, ">> edning an RSS feed");

} else if (tagName.equals(RssFeed.TAG_CHANNEL)) {

Log.e(TAG, "\t>> ending an Channel");

}

break;

}

default:

break;

}

eventType = parser.next();

}

Log.e(TAG, "end of document, it is over");

return parser.getEventType();

}

/**

* Parse a channel. MUST be start tag of an channel, otherwise exception thrown.

* Param XmlPullParser

* After calling this function, parser is positioned at END_TAG of Channel.

* return end tag of a channel

* @throws XmlPullParserException

* @throws ReaderBaseException

* @throws IOException

*/

private int parseChannel(XmlPullParser parser) throws XmlPullParserException, ReaderBaseException, IOException {

int eventType = parser.getEventType();

String tagName = parser.getName();

if (eventType != XmlPullParser.START_TAG || !RssFeed.TAG_CHANNEL.equals(tagName)) {

throw new ReaderBaseException("not start with 'start tag', is this a start of a channel?");

}

Log.e(TAG, "\tstarting " + tagName);

eventType = parser.nextTag();

while (eventType != XmlPullParser.END_TAG) {

switch (eventType) {

case XmlPullParser.START_TAG: {

final String tag = parser.getName();

if (tag.equals(RssFeed.TAG_IMAGE)) {

parseImage(parser);

} else if (tag.equals(RssFeed.TAG_ITEM)) {

parseItem(parser);

} else {

final String content = parser.nextText();

Log.e(TAG, tag + ": [" + content + "]");

}

// now it SHOULD be at END_TAG, ensure it

if (parser.getEventType() != XmlPullParser.END_TAG) {

throw new ReaderBaseException("not ending with 'end tag', did you finish parsing sub item?");

}

eventType = parser.nextTag();

break;

}

default:

break;

}

}

Log.e(TAG, "\tending " + parser.getName());

return parser.getEventType();

}

/**

* Parse image in a channel.

* Precondition: position must be at START_TAG and tag MUST be 'image'

* Postcondition: position is END_TAG of '/image'

* @throws IOException

* @throws XmlPullParserException

* @throws ReaderBaseException

*/

private int parseImage(XmlPullParser parser) throws XmlPullParserException, IOException, ReaderBaseException {

int eventType = parser.getEventType();

String tag = parser.getName();

if (eventType != XmlPullParser.START_TAG || !RssFeed.TAG_IMAGE.equals(tag)) {

throw new ReaderBaseException("not start with 'start tag', is this a start of an image?");

}

Log.e(TAG, "\t\tstarting image " + tag);

eventType = parser.nextTag();

while (eventType != XmlPullParser.END_TAG) {

switch (eventType) {

case XmlPullParser.START_TAG:

tag = parser.getName();

Log.e(TAG, tag + ": [" + parser.nextText() + "]");

// now it SHOULD be at END_TAG, ensure it

if (parser.getEventType() != XmlPullParser.END_TAG) {

throw new ReaderBaseException("not ending with 'end tag', did you finish parsing sub item?");

}

eventType = parser.nextTag();

break;

default:

break;

}

}

Log.e(TAG, "\t\tending image " + parser.getName());

return parser.getEventType();

}

/**

* Parse an item in a channel.

* Precondition: position must be at START_TAG and tag MUST be 'item'

* Postcondition: position is END_TAG of '/item'

* @throws IOException

* @throws XmlPullParserException

* @throws ReaderBaseException

*/

private int parseItem(XmlPullParser parser) throws XmlPullParserException, IOException, ReaderBaseException {

int eventType = parser.getEventType();

String tag = parser.getName();

if (eventType != XmlPullParser.START_TAG || !RssFeed.TAG_ITEM.equals(tag)) {

throw new ReaderBaseException("not start with 'start tag', is this a start of an item?");

}

Log.e(TAG, "\t\tstarting " + tag);

eventType = parser.nextTag();

while (eventType != XmlPullParser.END_TAG) {

switch (eventType) {

case XmlPullParser.START_TAG:

tag = parser.getName();

final String content = parser.nextText();

Log.e(TAG, tag + ": [" + content + "]");

// now it SHOULD be at END_TAG, ensure it

if (parser.getEventType() != XmlPullParser.END_TAG) {

throw new ReaderBaseException("not ending with 'end tag', did you finish parsing sub item?");

}

eventType = parser.nextTag();

break;

default:

break;

}

}

Log.e(TAG, "\t\tending " + parser.getName());

return parser.getEventType();

}

}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
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、付费专栏及课程。

余额充值