在Swift开发中,处理XML格式的数据是很多场景下的需求,比如接口返回XML格式数据、解析本地XML配置文件等。Foundation框架自带的XMLParser类配合XMLParserDelegate协议,可以高效完成XML数据的解析工作,不需要引入第三方库,兼容性和稳定性都有保障。

XMLParser与XMLParserDelegate基础概念
XMLParser是Foundation框架中用于解析XML数据的类,它采用事件驱动的解析方式,在解析过程中遇到不同的XML结构时会触发对应的回调方法。XMLParserDelegate是XMLParser的代理协议,开发者通过实现该协议的方法,就能获取解析过程中的节点信息、属性信息和文本内容。
使用XMLParser解析XML的核心流程分为三步:创建XMLParser实例、设置代理、调用解析方法启动解析。解析过程中,解析器会按照XML文档的顺序依次触发代理方法,开发者需要在代理方法中保存和处理需要的数据。
XMLParserDelegate核心方法详解
XMLParserDelegate提供了多个可选的回调方法,实际开发中常用的有以下几种:
- parserDidStartDocument:解析器开始解析XML文档时触发,适合在这里做初始化操作,比如创建存储解析结果的数组。
- parser(_:didStartElement:namespaceURI:qualifiedName:attributes:):遇到开始标签时触发,参数中包含标签名称、标签属性等信息。
- parser(_:foundCharacters:):遇到标签之间的文本内容时触发,返回当前标签内的字符串内容。
- parser(_:didEndElement:namespaceURI:qualifiedName:):遇到结束标签时触发,适合在这里处理当前标签解析完成后的逻辑,比如将临时存储的内容保存到结果集合中。
- parserDidEndDocument:解析器完成整个XML文档的解析时触发,适合在这里做解析收尾工作,比如通知上层解析完成。
- parser(_:parseErrorOccurred:):解析过程中出现错误时触发,可以在这里处理解析异常。
完整解析示例
下面以解析一个包含学生信息的XML数据为例,演示完整的解析过程。假设要解析的XML内容如下:
<students>
<student id="1">
<name>张三</name>
<age>20</age>
<major>计算机科学</major>
</student>
<student id="2">
<name>李四</name>
<age>21</age>
<major>软件工程</major>
</student>
</students>
首先定义存储学生信息的模型类:
// 学生模型
struct Student {
let id: String
let name: String
let age: String
let major: String
}
然后实现XMLParserDelegate协议,完成解析逻辑:
import Foundation
class StudentXMLParser: NSObject, XMLParserDelegate {
// 存储解析结果
private var students: [Student] = []
// 临时存储当前解析的学生信息
private var currentStudent: [String: String]?
// 临时存储当前解析的标签名
private var currentElement: String?
// 临时存储当前标签的文本内容
private var currentValue: String?
// 对外提供获取解析结果的方法
func parse(xmlData: Data) -> [Student] {
// 重置状态
students = []
currentStudent = nil
currentElement = nil
currentValue = nil
// 创建XMLParser实例
let parser = XMLParser(data: xmlData)
parser.delegate = self
// 启动解析
if parser.parse() {
print("XML解析成功")
} else {
print("XML解析失败")
}
return students
}
// 开始解析文档
func parserDidStartDocument(_ parser: XMLParser) {
print("开始解析XML文档")
}
// 遇到开始标签
func parser(_ parser: XMLParser, didStartElement elementName: String, namespaceURI: String?, qualifiedName qName: String?, attributes attributeDict: [String : String] = [:]) {
currentElement = elementName
if elementName == "student" {
// 遇到student开始标签,初始化临时学生信息,保存id属性
currentStudent = [:]
currentStudent?["id"] = attributeDict["id"]
}
}
// 遇到标签文本内容
func parser(_ parser: XMLParser, foundCharacters string: String) {
// 过滤掉空白字符和换行符
let trimmedString = string.trimmingCharacters(in: .whitespacesAndNewlines)
if !trimmedString.isEmpty {
currentValue = trimmedString
}
}
// 遇到结束标签
func parser(_ parser: XMLParser, didEndElement elementName: String, namespaceURI: String?, qualifiedName qName: String?) {
if elementName == "student" {
// 遇到student结束标签,将临时学生信息转为Student模型存入结果数组
if let studentInfo = currentStudent,
let id = studentInfo["id"],
let name = studentInfo["name"],
let age = studentInfo["age"],
let major = studentInfo["major"] {
let student = Student(id: id, name: name, age: age, major: major)
students.append(student)
}
currentStudent = nil
} else if let value = currentValue, var student = currentStudent {
// 其他标签结束,将文本内容存入临时学生信息
student[elementName] = value
currentStudent = student
currentValue = nil
}
currentElement = nil
}
// 解析完成
func parserDidEndDocument(_ parser: XMLParser) {
print("XML文档解析完成,共解析到(students.count)个学生")
}
// 解析错误
func parser(_ parser: XMLParser, parseErrorOccurred parseError: Error) {
print("解析出现错误:(parseError.localizedDescription)")
}
}
调用解析器的代码示例如下:
// 构造XML字符串
let xmlString = """
<students>
<student id="1">
<name>张三</name>
<age>20</age>
<major>计算机科学</major>
</student>
<student id="2">
<name>李四</name>
<age>21</age>
<major>软件工程</major>
</student>
</students>
"""
if let xmlData = xmlString.data(using: .utf8) {
let parser = StudentXMLParser()
let result = parser.parse(xmlData: xmlData)
// 打印解析结果
for student in result {
print("学生ID:(student.id),姓名:(student.name),年龄:(student.age),专业:(student.major)")
}
}
解析注意事项
在实际使用XMLParserDelegate解析XML时,需要注意以下几点:
- foundCharacters方法可能会被多次调用同一个标签的文本内容,比如文本内容过长时解析器会分段返回,因此需要在遇到结束标签时再统一处理当前标签的文本内容,避免内容拼接错误。
- XML中的空白字符和换行符也会被foundCharacters方法返回,需要在处理文本内容时过滤掉无用的空白字符,避免影响数据准确性。
- 如果XML结构比较复杂,包含多层嵌套,需要在代理方法中维护一个标签栈,记录当前的解析层级,避免不同层级的同名标签数据混淆。
- 解析过程中如果出现错误,parse方法会返回false,同时触发parseErrorOccurred回调,需要做好错误处理的逻辑。
常见问题解答
解析中文内容出现乱码怎么办
需要确保传入XMLParser的Data是用正确的编码格式生成的,通常XML文档使用UTF-8编码,只要生成Data时使用UTF-8编码就不会出现乱码问题。
如何解析带命名空间的XML
XMLParserDelegate的方法参数中包含了namespaceURI和qualifiedName参数,如果需要处理带命名空间的XML,可以通过这两个参数判断标签的命名空间,区分不同命名空间下的同名标签。
解析大文件XML会不会占用太多内存
XMLParser是流式解析,不会一次性把整个XML文档加载到内存中,而是边解析边触发回调,因此即使解析很大的XML文件,内存占用也比较低,适合处理大体积XML数据。
SwiftXMLParserDelegateFoundationXML解析修改时间:2026-06-14 22:03:39