0

0

Groovy XmlSlurper vs XmlParser 在处理上传文件时的差异

星降

星降

发布时间:2026-01-25 07:27:09

|

718人浏览过

|

来源于php中文网

原创

XmlSlurper 无法直接解析 multipart/form-data 请求体,因其含 boundary 干扰;应先用 getPart() 提取纯 XML 字段再解析,Slurper 适合松散配置类 XML,Parser 适合需 DTD/Schema 校验的严格 XML。

groovy xmlslurper vs xmlparser 在处理上传文件时的差异

XmlSlurper 无法解析 multipart/form-data 中的原始 XML 内容

上传文件时,HTTP 请求体是 multipart/form-data 编码,XML 数据通常作为某个表单字段(如 xmlPayload)的值存在,而非独立文档。XmlSlurper 默认期望输入是完整、格式良好的 XML 字符串或 InputStream,但若你直接把整个 request.getInputStream() 传给它,会遇到边界分隔符(boundary)干扰,导致 org.xml.sax.SAXParseException:*Content is not allowed in prolog*。

实操建议:

通义灵码
通义灵码

阿里云出品的一款基于通义大模型的智能编码辅助工具,提供代码智能生成、研发智能问答能力

下载
  • 先用 request.getPart("xmlPayload")(Servlet 3.0+)或第三方库(如 Apache Commons FileUpload)提取出纯 XML 字符串或 InputStream
  • 确保提取后的内容以 或根元素开头,无前导空格、换行或 multipart 头部
  • XmlSlurper 对空白和命名空间较宽容,适合快速读取结构松散的配置类 XML

XmlParser 更适合校验严格、含 DTD/Schema 的上传 XML

XmlParser 使用标准 SAX/DOM 解析器,能识别并报告 DTD 声明、实体引用、命名空间约束等细节。如果你接收的是带 的行业标准 XML(如 UBL 发票、FHIR 医疗数据),XmlParser 能提前暴露 org.xml.sax.SAXParseException: Element type "xxx" must be declared 类错误,而 XmlSlurper 会静默忽略或抛出更模糊的异常。

实操建议:

  • new XmlParser(false, true) 禁用 DTD 加载(防 XXE),但保留命名空间支持
  • 配合 try/catch SAXParseException 捕获结构问题,并返回 HTTP 400 + 具体错误位置(e.lineNumber, e.columnNumber
  • 避免在解析前调用 toString() 转成字符串再解析——这会丢失原始编码信息,易触发 Invalid byte 1 of 1-byte UTF-8 sequence

内存占用与流式处理的关键区别

XmlSlurper 是懒加载的:它不立即构建完整对象树,而是返回一个代理(GPathResult),只有访问属性或调用 text() 时才解析对应节点。XmlParser 则默认一次性将整个文档加载为内存中的 DOM 树。这对大文件上传很关键——比如上传 10MB 的 XML 报文,XmlParser 可能直接触发 OutOfMemoryError,而 XmlSlurper 配合流式提取可控制内存峰值。

实操建议:

  • 对大 XML 文件,优先用 new XmlSlurper(false).parse(new BufferedInputStream(part.inputStream)),禁用命名空间简化开销
  • 不要对 Slurper 结果做 collectEntries{[it.@id, it]} 全量转 Map——这会强制遍历全部节点,失去懒加载优势
  • 若需流式验证(如只检查根元素名和版本号),用 XmlParser 的 setFeature("http://apache.org/xml/features/nonvalidating/load-external-dtd", false) + 自定义 DefaultHandler,比两者都轻量
def xmlStr = ''
// XmlSlurper:返回 GPathResult,访问时才解析
def slurper = new XmlSlurper().parseText(xmlStr)
println slurper.@id // OK,输出 123

// XmlParser:立即构建完整节点树
def parser = new XmlParser().parseText(xmlStr)
println parser.@id // 同样 OK,但整棵树已驻留内存
XmlSlurper 和 XmlParser 的差异不在“能不能解析上传的 XML”,而在于你是否需要它立刻报错、能否承受全量内存驻留、以及原始内容是否已被干净剥离。多数 Web 场景下,先用标准方式提取字段值,再按 XML 严格性要求选解析器——而不是让解析器去对抗 multipart 边界。

相关专题

更多
servlet生命周期
servlet生命周期

Servlet生命周期是指Servlet从创建到销毁的整个过程。本专题为大家提供servlet生命周期的各类文章,大家可以免费体验。

374

2023.08.08

pdf怎么转换成xml格式
pdf怎么转换成xml格式

将 pdf 转换为 xml 的方法:1. 使用在线转换器;2. 使用桌面软件(如 adobe acrobat、itext);3. 使用命令行工具(如 pdftoxml)。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

1897

2024.04.01

xml怎么变成word
xml怎么变成word

步骤:1. 导入 xml 文件;2. 选择 xml 结构;3. 映射 xml 元素到 word 元素;4. 生成 word 文档。提示:确保 xml 文件结构良好,并预览 word 文档以验证转换是否成功。想了解更多xml的相关内容,可以阅读本专题下面的文章。

2089

2024.08.01

xml是什么格式的文件
xml是什么格式的文件

xml是一种纯文本格式的文件。xml指的是可扩展标记语言,标准通用标记语言的子集,是一种用于标记电子文件使其具有结构性的标记语言。想了解更多相关的内容,可阅读本专题下面的相关文章。

1043

2024.11.28

js 字符串转数组
js 字符串转数组

js字符串转数组的方法:1、使用“split()”方法;2、使用“Array.from()”方法;3、使用for循环遍历;4、使用“Array.split()”方法。本专题为大家提供js字符串转数组的相关的文章、下载、课程内容,供大家免费下载体验。

278

2023.08.03

js截取字符串的方法
js截取字符串的方法

js截取字符串的方法有substring()方法、substr()方法、slice()方法、split()方法和slice()方法。本专题为大家提供字符串相关的文章、下载、课程内容,供大家免费下载体验。

212

2023.09.04

java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1493

2023.10.24

字符串介绍
字符串介绍

字符串是一种数据类型,它可以是任何文本,包括字母、数字、符号等。字符串可以由不同的字符组成,例如空格、标点符号、数字等。在编程中,字符串通常用引号括起来,如单引号、双引号或反引号。想了解更多字符串的相关内容,可以阅读本专题下面的文章。

622

2023.11.24

c++ 根号
c++ 根号

本专题整合了c++根号相关教程,阅读专题下面的文章了解更多详细内容。

41

2026.01.23

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
RunnerGo从入门到精通
RunnerGo从入门到精通

共22课时 | 1.7万人学习

尚学堂Mahout视频教程
尚学堂Mahout视频教程

共18课时 | 3.2万人学习

Linux优化视频教程
Linux优化视频教程

共14课时 | 3.1万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号