推荐使用XmlPullParser,因其轻量高效、内存占用低且为Android官方支持;相比DOM(易OOM)和SAX(回调嵌套深),它采用拉模式,配合Kotlin的when和作用域函数更简洁可控。
Kotlin 处理 Android 中的 XML 数据,推荐使用 XmlPullParser(XML Pull Parser),它是 Android 官方支持、轻量高效、内存占用低的流式解析器,特别适合移动设备。
DOM 会把整个 XML 加载进内存,对大文件容易 OOM;SAX 是事件驱动但回调嵌套深、Kotlin 写起来不够直观。XmlPullParser 是“拉模式”:你主动调用 next() 或 nextTag() 推动解析,逻辑清晰、易控制、可随时中断,配合 Kotlin 的 when 和作用域函数写起来很简洁。
从资源文件(如 res/xml/config.xml)或输入流中获取解析器实例:
// 示例:解析 res/xml/items.xml
val parser = resources.getXml(R.xml.items) parser.setFeature(XmlPullParser.FEATURE_PROCESS_NAMESPACES, false)var eventType = parser.eventType while (eventType != XmlPullParser.END_DOCUMENT) { when (eventType) { XmlPullParser.START_TAG -> { when (parser.name) {
"item" -> { val id = parser.getAttributeValue(null, "id") val name = parser.nextText() // 直接读取该标签的文本内容 // 处理 item 对象... } "category" -> { val type = parser.getAttributeValue(null, "type") val content = parser.nextText() } } } XmlPullParser.TEXT -> { / 可选:处理纯文本节点 / } XmlPullParser.END_TAG -> { / 标签闭合时可做收尾 / } } eventType = parser.next() }
xmlns="http://example.com"),需用 parser.namespace 或传 null 获取无前缀属性,否则 getAttributeValue(null, "attr") 可能返回 nullnextTag() 简化结构:它自动跳过 TEXT 和 COMMENT,只停在 START_TAG 或 END_TAG,适合已知层级结构的 XMLnextText():它隐含执行一次 next() 并期望下一个事件是 TEXT;若实际是 END_TAG,会抛 XmlPullParserException。建议先判断 parser.eventType == XmlPullParser.TEXT 再取值XmlPullParserException 和 IOException 都要 try-catch,尤其网络加载 XML 时 IO 不稳定可以为 XmlPullParser 添加 Kotlin 扩展,让常见操作更安全简洁:
fun XmlPullParser.require(name: String, ns: String? = null) {
if (eventPosition != START_TAG || name != this.name || ns != this.namespace) {
throw XmlPullParserException("Expected $name but was ${this.name}")
}
}
// 使用示例:
parser.require("items")
while (parser.nextTag() == XmlPullParser.START_TAG) {
when (parser.name) {
"item" -> parseItem(parser)
else -> parser.skip() // 跳过未知标签
}
}
// skip() 是 Android API 21+ 提供的便捷方法,自动跳过整个子树
不复杂但容易忽略细节,掌握好事件流转和边界判断,XmlPullParser 在 Kotlin 里既可靠又干净。