0

0

使用RestTemplate获取XML对象列表为空问题的解决方案

DDD

DDD

发布时间:2025-10-04 16:26:01

|

449人浏览过

|

来源于php中文网

原创

使用RestTemplate获取XML对象列表为空问题的解决方案

本文详细探讨了在使用Spring RestTemplate结合Jackson处理外部API返回的XML列表时,可能遇到的空列表问题。核心在于通过Jackson的XML特定注解(如@JacksonXmlRootElement、@JacksonXmlProperty和@JacksonXmlElementWrapper)正确映射XML结构到Java POJO,尤其是在处理非包装(unwrapped)列表元素时,确保数据能够被正确反序列化。文章提供了完整的代码示例和配置说明,帮助开发者有效解决此类XML数据绑定难题。

1. 引言:RestTemplate与XML数据绑定挑战

在使用spring框架的resttemplate与外部api进行交互时,如果api返回的是xml格式的数据,并且数据中包含一个对象列表,开发者可能会遇到列表为空的问题。这通常不是resttemplate本身的问题,而是jackson xml数据绑定库(jackson-dataformat-xml)在将xml结构反序列化为java对象(pojo)列表时,未能正确识别和映射xml元素。本教程将深入分析此类问题,并提供一套基于jackson注解的完整解决方案。

2. 问题分析:XML列表反序列化失败的常见原因

假设我们从外部API获取的XML数据结构如下:


    
        1
        Item A
        Value1A
        Value2A
        Value3A
    
    
       2
       Item B
       Value1B
       Value2B
       Value3B
   
   

当尝试将这种XML结构映射到Java POJO列表时,常见的错误做法可能包括:

  • POJO字段与XML元素名称不匹配: Java类中的字段名未与XML元素名保持一致,或未通过注解明确指定映射关系。
  • 列表包装器注解使用不当: 对于像......这种“非包装”列表(即列表元素直接作为父元素的子元素,没有额外的列表包装标签),@JacksonXmlElementWrapper注解的配置至关重要。
  • 缺少必要的Jackson XML注解: 未使用@JacksonXmlRootElement或@JacksonXmlProperty来指导Jackson如何解析XML结构。
  • @JsonIgnoreProperties误用: @JsonIgnoreProperties主要用于JSON反序列化,在XML场景下,其行为可能不如预期,或者不适用于解决列表为空的问题。对于XML,通常是未注解的字段会被忽略,除非设置了严格的失败策略。

3. 解决方案:使用Jackson XML注解正确映射

解决此问题的关键在于正确使用jackson-dataformat-xml提供的注解来精确指导Jackson如何将XML数据反序列化为Java对象。

3.1 核心依赖

首先,确保pom.xml中包含了Jackson XML数据绑定的相关依赖:


    com.fasterxml.jackson.dataformat
    jackson-dataformat-xml


    javax.xml.bind
    jaxb-api
    2.3.1 

jaxb-api在Java 9及更高版本中被移除出JDK默认模块,因此对于现代Spring Boot应用,通常需要显式添加此依赖。

3.2 定义子对象POJO (Object类)

针对XML中的元素,我们需要创建一个对应的Java类。为了确保Jackson能正确识别每个字段,应使用@JacksonXmlRootElement标记根元素,并使用@JacksonXmlProperty标记每个字段。

package it.me.model;

import com.fasterxml.jackson.dataformat.xml.annotation.JacksonXmlProperty;
import com.fasterxml.jackson.dataformat.xml.annotation.JacksonXmlRootElement;
import java.io.Serializable;

@JacksonXmlRootElement(localName = "Object") // 明确指定XML根元素名为"Object"
public class Object implements Serializable {

    private static final long serialVersionUID = 1L;

    @JacksonXmlProperty // 默认映射到同名字段
    private String id;

    @JacksonXmlProperty
    private String name;

    @JacksonXmlProperty
    private String var1;

    @JacksonXmlProperty
    private String var2;

    @JacksonXmlProperty
    private String var3;

    // 无参构造函数是Jackson反序列化的必要条件
    public Object() {}

    // 全参构造函数(可选,但推荐)
    public Object(String id, String name, String var1, String var2, String var3) {
        this.id = id;
        this.name = name;
        this.var1 = var1;
        this.var2 = var2;
        this.var3 = var3;
    }

    // Getters and Setters (Jackson需要它们来访问和设置字段值)
    public String getId() { return id; }
    public void setId(String id) { this.id = id; }
    public String getName() { return name; }
    public void setName(String name) { this.name = name; }
    public String getVar1() { return var1; }
    public void setVar1(String var1) { this.var1 = var1; }
    public String getVar2() { return var2; }
    public void setVar2(String var2) { this.var2 = var2; }
    public String getVar3() { return var3; }
    public void setVar3(String var3) { this.var3 = var3; }

    // toString, equals, hashCode 等方法(推荐重写,便于调试和使用)
    @Override
    public String toString() {
        return "Object{" +
               "id='" + id + '\'' +
               ", name='" + name + '\'' +
               ", var1='" + var1 + '\'' +
               ", var2='" + var2 + '\'' +
               ", var3='" + var3 + '\'' +
               '}';
    }
}

注意事项:

InsCode
InsCode

InsCode 是CSDN旗下的一个无需安装的编程、协作和分享社区

下载
  • @JacksonXmlRootElement(localName = "Object"):明确指出这个Java类对应XML中的标签。
  • @JacksonXmlProperty:用于标记需要从XML元素中映射的字段。如果XML元素名与Java字段名一致,则无需指定localName;否则,需要@JacksonXmlProperty(localName = "xmlElementName")。
  • 所有需要映射的XML字段都应在POJO中定义并使用@JacksonXmlProperty进行注解。如果XML中存在POJO中未定义的字段,Jackson默认会忽略它们(除非配置了DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES)。如果只想获取部分字段,依然需要确保这些字段被正确注解。

3.3 定义列表包装器POJO (ArrayOfObject类)

针对包含列表的元素,我们需要定义一个包装类。关键在于正确处理列表字段的反序列化。

package it.me.model;

import com.fasterxml.jackson.dataformat.xml.annotation.JacksonXmlElementWrapper;
import com.fasterxml.jackson.dataformat.xml.annotation.JacksonXmlProperty;
import com.fasterxml.jackson.dataformat.xml.annotation.JacksonXmlRootElement;
import java.io.Serializable;
import java.util.ArrayList;
import java.util.List;

@JacksonXmlRootElement(localName = "ArrayOfObject") // 明确指定XML根元素名为"ArrayOfObject"
public class ArrayOfObject implements Serializable {

    private static final long serialVersionUID = 1L;

    // 关键注解:
    // @JacksonXmlProperty(localName = "Object"):指定列表中的每个元素对应的XML标签名是"Object"
    // @JacksonXmlElementWrapper(useWrapping = false):告诉Jackson,XML中没有额外的列表包装标签,
    //                                                元素直接作为的子元素。
    @JacksonXmlProperty(localName = "Object")
    @JacksonXmlElementWrapper(useWrapping = false)
    private List objects = new ArrayList<>(); // 字段名可以自由命名,但要通过注解正确映射

    public ArrayOfObject() {}

    public ArrayOfObject(List objects) {
        this.objects = objects;
    }

    // Getters and Setters
    public List getObjects() { return objects; }
    public void setObjects(List objects) { this.objects = objects; }

    @Override
    public String toString() {
        return "ArrayOfObject{" +
               "objects=" + objects +
               '}';
    }
}

注意事项:

  • @JacksonXmlRootElement(localName = "ArrayOfObject"):指定这个Java类对应XML中的标签。
  • @JacksonXmlProperty(localName = "Object"):此注解应用于列表字段objects,它告诉Jackson,当解析ArrayOfObject时,寻找名为Object的子元素来填充此列表。
  • @JacksonXmlElementWrapper(useWrapping = false):这是解决“空列表”问题的核心。它指示Jackson,XML结构中没有一个额外的标签来“包装”列表中的所有Object元素。例如,如果XML是......,那么useWrapping就应该设置为true,并且localName设置为Objects。但在当前示例中,直接是的子元素,因此必须设置为false。

3.4 RestTemplate调用

最后,在Spring Controller或Service中,使用RestTemplate进行API调用时,将期望的反序列化类型指定为包装类ArrayOfObject.class。

package it.me.controller;

import it.me.model.ArrayOfObject;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
import org.springframework.web.client.RestTemplate;
import java.util.List;

@RestController
@RequestMapping("/api") // 建议为Controller添加一个基础路径
public class RestCallController {

    private final String EXTERNAL_API_URI = "http://localhost:8080/external/objects"; // 替换为实际的外部API地址

    @GetMapping("/objects")
    public List getObjectsFromExternalApi() {
        RestTemplate restTemplate = new RestTemplate();
        // 直接将XML反序列化为ArrayOfObject类型
        ArrayOfObject resultWrapper = restTemplate.getForObject(EXTERNAL_API_URI, ArrayOfObject.class);

        // 返回包装类中的列表
        return resultWrapper != null ? resultWrapper.getObjects() : new ArrayList<>();
    }
}

注意事项:

  • RestTemplate会自动检测响应的Content-Type。如果响应是application/xml,并且jackson-dataformat-xml依赖已存在,RestTemplate会使用Jackson的XML消息转换器进行反序列化。
  • getForObject(uri, ArrayOfObject.class):指定了RestTemplate应将响应体反序列化为ArrayOfObject的实例。

4. 总结与最佳实践

通过上述步骤,我们可以成功地使用RestTemplate和Jackson将外部XML API返回的列表数据反序列化为Java POJO。解决此类问题的关键在于:

  1. 引入正确的Jackson XML依赖 (jackson-dataformat-xml和jaxb-api)。
  2. 为每个XML元素定义对应的Java POJO,并使用@JacksonXmlRootElement和@JacksonXmlProperty进行精确映射。
  3. 特别注意列表的反序列化
    • 在包含列表的POJO中,使用@JacksonXmlProperty(localName = "ElementNameOfListItem")来指定列表中每个元素的XML标签名。
    • 根据XML结构,正确配置@JacksonXmlElementWrapper(useWrapping = false)或@JacksonXmlElementWrapper(useWrapping = true, localName = "WrapperElementName")。对于本例中的非包装列表,useWrapping = false是核心。
  4. 提供无参构造函数和完整的Getter/Setter方法,这是Jackson进行反序列化的基本要求。
  5. 在RestTemplate调用中,将目标类型指定为包含列表的包装类

遵循这些最佳实践,可以有效避免在使用RestTemplate和Jackson处理XML列表时遇到的反序列化问题,确保数据能够被准确无误地解析和使用。

相关专题

更多
java
java

Java是一个通用术语,用于表示Java软件及其组件,包括“Java运行时环境 (JRE)”、“Java虚拟机 (JVM)”以及“插件”。php中文网还为大家带了Java相关下载资源、相关课程以及相关文章等内容,供大家免费下载使用。

841

2023.06.15

java正则表达式语法
java正则表达式语法

java正则表达式语法是一种模式匹配工具,它非常有用,可以在处理文本和字符串时快速地查找、替换、验证和提取特定的模式和数据。本专题提供java正则表达式语法的相关文章、下载和专题,供大家免费下载体验。

742

2023.07.05

java自学难吗
java自学难吗

Java自学并不难。Java语言相对于其他一些编程语言而言,有着较为简洁和易读的语法,本专题为大家提供java自学难吗相关的文章,大家可以免费体验。

737

2023.07.31

java配置jdk环境变量
java配置jdk环境变量

Java是一种广泛使用的高级编程语言,用于开发各种类型的应用程序。为了能够在计算机上正确运行和编译Java代码,需要正确配置Java Development Kit(JDK)环境变量。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

397

2023.08.01

java保留两位小数
java保留两位小数

Java是一种广泛应用于编程领域的高级编程语言。在Java中,保留两位小数是指在进行数值计算或输出时,限制小数部分只有两位有效数字,并将多余的位数进行四舍五入或截取。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

399

2023.08.02

java基本数据类型
java基本数据类型

java基本数据类型有:1、byte;2、short;3、int;4、long;5、float;6、double;7、char;8、boolean。本专题为大家提供java基本数据类型的相关的文章、下载、课程内容,供大家免费下载体验。

446

2023.08.02

java有什么用
java有什么用

java可以开发应用程序、移动应用、Web应用、企业级应用、嵌入式系统等方面。本专题为大家提供java有什么用的相关的文章、下载、课程内容,供大家免费下载体验。

430

2023.08.02

java在线网站
java在线网站

Java在线网站是指提供Java编程学习、实践和交流平台的网络服务。近年来,随着Java语言在软件开发领域的广泛应用,越来越多的人对Java编程感兴趣,并希望能够通过在线网站来学习和提高自己的Java编程技能。php中文网给大家带来了相关的视频、教程以及文章,欢迎大家前来学习阅读和下载。

16926

2023.08.03

Java JVM 原理与性能调优实战
Java JVM 原理与性能调优实战

本专题系统讲解 Java 虚拟机(JVM)的核心工作原理与性能调优方法,包括 JVM 内存结构、对象创建与回收流程、垃圾回收器(Serial、CMS、G1、ZGC)对比分析、常见内存泄漏与性能瓶颈排查,以及 JVM 参数调优与监控工具(jstat、jmap、jvisualvm)的实战使用。通过真实案例,帮助学习者掌握 Java 应用在生产环境中的性能分析与优化能力。

19

2026.01.20

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 2.7万人学习

C# 教程
C# 教程

共94课时 | 7.1万人学习

Java 教程
Java 教程

共578课时 | 48.1万人学习

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

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