0

0

使用 RestTemplate 获取并处理外部API数据:去重与重塑

霞舞

霞舞

发布时间:2025-09-28 10:29:41

|

751人浏览过

|

来源于php中文网

原创

使用 resttemplate 获取并处理外部api数据:去重与重塑

本文详细介绍了如何使用 Spring RestTemplate 从外部 API 获取数据,并对获取到的数据进行处理,包括去重、过滤以及将原始数据结构转换为更适合自身 API 响应的自定义数据传输对象(DTO)。通过示例代码,演示了如何结合 Java Stream API 实现高效的数据处理流程,确保输出数据的准确性和一致性,从而提升API的可用性。

1. 引言:RestTemplate与外部数据集成

在现代微服务架构中,应用程序经常需要与各种外部服务进行交互,获取数据或调用功能。Spring Framework 提供了 RestTemplate 作为一种便捷的方式来执行 HTTP 请求。然而,从外部 API 获取的数据往往需要经过一系列的清洗、转换和过滤,才能符合我们自身 API 的设计要求。本教程将以一个具体的案例为例,讲解如何利用 RestTemplate 获取数据,并进一步对其进行去重和结构重塑,最终以自定义的格式暴露给前端

2. 原始数据结构与需求分析

假设我们正在构建一个服务,需要从一个外部天气API获取省份数据。原始数据结构如下:

Provinces 类

import com.fasterxml.jackson.annotation.JsonIgnoreProperties;
import com.fasterxml.jackson.annotation.JsonProperty;
import java.util.List;

@JsonIgnoreProperties(ignoreUnknown = true)
public class Provinces {

    @JsonProperty("provincial")
    private List provinces;

    public Provinces() {}

    public Provinces(List provinces) {
        this.provinces = provinces;
    }

    @JsonProperty("provincial")
    public List getprovinces() {
        return provinces;
    }

    @JsonProperty("Test") // 示例中存在此注解,实际应修正为正确的setter注解
    public void setprovinces(List provinces) {
        this.provinces = provinces;
    }
}

ProvincesData 类

import com.fasterxml.jackson.annotation.JsonIgnoreProperties;
import com.fasterxml.jackson.annotation.JsonProperty;

@JsonIgnoreProperties(ignoreUnknown = true)
public class ProvincesData {

    @JsonProperty("CODPROV")
    private String codProv;

    @JsonProperty("NOMBRE_PROVINCIA")
    private String nomeProvincia;

    @JsonProperty("CODAUTON")
    private String codAuton;

    @JsonProperty("COMUNIDAD_CIUDAD_AUTONOMA")
    private String comunidadeCidadeAutonoma;

    public ProvincesData() {}

    public ProvincesData(String codProv, String nomeProvincia, String codAuton, String comunidadeCidadeAutonoma) {
        this.codProv = codProv;
        this.nomeProvincia = nomeProvincia;
        this.codAuton = codAuton;
        this.comunidadeCidadeAutonoma = comunidadeCidadeAutonoma; 
    }

    @JsonProperty("CODPROV")
    public String getCodProv() {
        return codProv;
    }

    @JsonProperty("Test") // 示例中存在此注解,实际应修正为正确的setter注解
    public void setCodProv(String codProv) {
        this.codProv = codProv;
    }

    // 省略其他属性的getter和setter,确保它们是正确的
    @JsonProperty("NOMBRE_PROVINCIA")
    public String getNomeProvincia() {
        return nomeProvincia;
    }

    public void setNomeProvincia(String nomeProvincia) {
        this.nomeProvincia = nomeProvincia;
    }

    @JsonProperty("CODAUTON")
    public String getCodAuton() {
        return codAuton;
    }

    public void setCodAuton(String codAuton) {
        this.codAuton = codAuton;
    }

    @JsonProperty("COMUNIDAD_CIUDAD_AUTONOMA")
    public String getComunidadeCidadeAutonoma() {
        return comunidadeCidadeAutonoma;
    }

    public void setComunidadeCidadeAutonoma(String comunidadeCidadeAutonoma) {
        this.comunidadeCidadeAutonoma = comunidadeCidadeAutonoma;
    }
}

我们的核心需求是:

  1. 从外部API获取 Provinces 对象,其中包含 List
  2. 对 List 进行过滤,移除基于 codAuton 字段重复的数据,只保留唯一的自治区编码
  3. 将过滤后的数据转换为一个新的列表结构,其中只包含 codAuton 和 comunidadeCidadeAutonoma 字段,并且可能需要重命名这些字段以适应我们自己的API规范。

3. 使用 RestTemplate 获取原始数据

首先,我们需要一个辅助方法来使用 RestTemplate 调用外部 API 并获取原始数据。

import org.springframework.web.client.RestTemplate;
import java.util.ArrayList;
import java.util.List;
import java.util.stream.Collectors;

public class Templates {

    public static Provinces restTemplateProvince(RestTemplate restTemplate) {
        String provinceCommunityURL = "https://www.el-tiempo.net/api/json/v2/provincias";
        Provinces province = restTemplate.getForObject(provinceCommunityURL, Provinces.class);
        return province;
    }
}

在实际应用中,RestTemplate 实例通常通过 Spring 的依赖注入机制进行管理。

4. 核心处理:数据去重与过滤

获取到原始 Provinces 对象后,下一步是处理其内部的 List,以移除基于 codAuton 字段的重复项。我们可以利用 Java 8 的 Stream API 结合一个辅助 Set(或 List)来实现高效去重。

以下是优化后的 restTemplateProvince 方法,它在获取数据后立即执行去重逻辑:

import org.springframework.web.client.RestTemplate;
import java.util.ArrayList;
import java.util.HashSet; // 推荐使用HashSet进行更高效的去重判断
import java.util.List;
import java.util.Set;
import java.util.stream.Collectors;

public class Templates {

    public static Provinces restTemplateProvince(RestTemplate restTemplate) {
        String provinceCommunityURL = "https://www.el-tiempo.net/api/json/v2/provincias";
        Provinces province = restTemplate.getForObject(provinceCommunityURL, Provinces.class);

        if (province != null && province.getprovinces() != null) {
            // 使用 HashSet 比 ArrayList 更高效地检查元素是否存在
            Set includedCodAuton = new HashSet<>(); 

            List filteredProvinces = province.getprovinces()
                .stream()
                .filter(p -> {
                    // 如果 codAuton 已经存在于 Set 中,则过滤掉此项(返回false)
                    // 否则,将其添加到 Set 中并保留此项(返回true)
                    return includedCodAuton.add(p.getCodAuton());
                })
                .collect(Collectors.toList());

            province.setprovinces(filteredProvinces);
        }
        return province;
    }
}

去重逻辑说明:

Designs.ai
Designs.ai

AI设计工具

下载
  • 我们创建了一个 HashSet includedCodAuton 来存储已经处理过的 codAuton 值。HashSet 提供了 O(1) 的平均时间复杂度来检查元素是否存在和添加元素,这比 ArrayList.contains() 的 O(n) 效率更高。
  • stream().filter() 方法遍历 ProvincesData 列表。
  • 在 filter 的 lambda 表达式中,includedCodAuton.add(p.getCodAuton()) 会尝试将当前 ProvincesData 对象的 codAuton 添加到 Set 中。
    • 如果 codAuton 之前不存在于 Set 中,add() 方法会返回 true,表示添加成功,当前 ProvincesData 对象会被保留。
    • 如果 codAuton 已经存在于 Set 中,add() 方法会返回 false,表示添加失败(因为 Set 不允许重复元素),当前 ProvincesData 对象会被过滤掉。
  • 最终,collect(Collectors.toList()) 将过滤后的元素收集成一个新的 List

5. 数据转换与重塑:生成自定义API响应

仅仅去重还不足以满足我们的所有需求。我们还需要将数据结构重塑,只包含 codAuton 和 comunidadeCidadeAutonoma 字段,并且可能希望重命名这些字段。最专业的做法是定义一个专门用于 API 响应的数据传输对象 (DTO)。

定义 AutonomyRegionDTO:

// AutonomyRegionDTO.java
public class AutonomyRegionDTO {
    private String regionCode; // 对应原始的 codAuton
    private String regionName; // 对应原始的 comunidadeCidadeAutonoma

    public AutonomyRegionDTO(String regionCode, String regionName) {
        this.regionCode = regionCode;
        this.regionName = regionName;
    }

    // Getters and Setters
    public String getRegionCode() {
        return regionCode;
    }

    public void setRegionCode(String regionCode) {
        this.regionCode = regionCode;
    }

    public String getRegionName() {
        return regionName;
    }

    public void setRegionName(String regionName) {
        this.regionName = regionName;
    }
}

现在,我们可以在 ProvinceService 中执行数据获取、去重和转换的完整流程。

改造 ProvinceService:

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import org.springframework.web.client.RestTemplate;
import java.util.List;
import java.util.stream.Collectors;

@Service
public class ProvinceService {

    @Autowired
    private RestTemplate restTemplate; // 确保 RestTemplate 已被正确配置和注入

    public List getUniqueAutonomyRegions() {
        Provinces rawProvinces = Templates.restTemplateProvince(restTemplate); // 调用包含去重逻辑的方法

        if (rawProvinces != null && rawProvinces.getprovinces() != null) {
            // 将过滤后的 ProvincesData 列表映射到 AutonomyRegionDTO 列表
            return rawProvinces.getprovinces().stream()
                .map(pData -> new AutonomyRegionDTO(pData.getCodAuton(), pData.getComunidadeCidadeAutonoma()))
                .collect(Collectors.toList());
        }
        return List.of(); // 返回空列表或抛出异常,根据业务需求而定
    }
}

map 操作说明:

  • 在去重后的 ProvincesData 列表上,我们继续使用 stream().map() 方法。
  • map 方法将每个 ProvincesData 对象转换为一个新的 AutonomyRegionDTO 对象,只提取并重命名了我们关心的字段。
  • collect(Collectors.toList()) 将转换后的 DTO 收集成一个 List

6. API层暴露处理后的数据

最后,我们的 RestController 将调用 ProvinceService 来获取处理后的数据,并将其作为 API 响应返回。

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RestController;
import java.util.List;

@RestController
public class ShowcaseController {

    @Autowired
    private ProvinceService provinceService;

    @GetMapping("/autonomy-regions")
    public List getAutonomyRegions(){
      return provinceService.getUniqueAutonomyRegions(); 
    }
}

现在,当访问 /autonomy-regions 接口时,将返回一个只包含去重后的自治区编码和名称的列表,并且字段名也已根据 AutonomyRegionDTO 的定义进行了重命名。

7. 注意事项与最佳实践

  • RestTemplate 配置:确保 RestTemplate 在 Spring 应用程序中正确配置并作为 Bean 注入。通常推荐使用 RestTemplateBuilder 来创建和配置 RestTemplate 实例,例如添加拦截器、消息转换器等。
  • 错误处理:在实际应用中,restTemplate.getForObject() 调用可能会因为网络问题、API 不可用或响应格式不正确而抛出异常。应添加 try-catch 块或使用 ResponseEntity 来处理这些潜在错误,提供健壮的用户体验。
  • 性能考量:对于非常大的数据集,虽然 HashSet 提供了高效的去重,但整个 Stream 处理过程仍需要将数据加载到内存。如果数据量极其庞大,可能需要考虑分页、外部存储或更高级的数据处理框架。
  • DTO 的重要性:使用 DTO(Data Transfer Object)是 API 设计中的一个最佳实践。它允许你将内部数据模型与外部 API 响应解耦,确保 API 响应的稳定性和安全性,同时可以根据客户端需求定制数据结构。
  • 关注点分离:将数据获取、去重、转换等逻辑分别封装在不同的方法或服务中,可以提高代码的可读性、可维护性和可测试性。
  • JSON 属性映射:@JsonProperty 注解用于将 JSON 字段名映射到 Java 对象的属性名。请确保在 ProvincesData 和 Provinces 类中的 setter 方法上不要使用错误的 @JsonProperty("Test") 注解,这可能导致反序列化问题。通常,@JsonProperty 只用于 getter 或字段本身。

8. 总结

本教程演示了如何利用 Spring RestTemplate 从外部 API 获取数据,并通过 Java Stream API 结合 HashSet 实现高效的数据去重。更重要的是,我们展示了如何将原始的、可能冗余的数据结构转换为一个更精简、更符合自身 API 需求的自定义 DTO 列表。通过这种方式,我们不仅解决了数据处理的实际问题,也遵循了良好的软件设计原则,如关注点分离和使用 DTO 进行数据传输。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
spring框架介绍
spring框架介绍

本专题整合了spring框架相关内容,想了解更多详细内容,请阅读专题下面的文章。

114

2025.08.06

Java Spring Security 与认证授权
Java Spring Security 与认证授权

本专题系统讲解 Java Spring Security 框架在认证与授权中的应用,涵盖用户身份验证、权限控制、JWT与OAuth2实现、跨站请求伪造(CSRF)防护、会话管理与安全漏洞防范。通过实际项目案例,帮助学习者掌握如何 使用 Spring Security 实现高安全性认证与授权机制,提升 Web 应用的安全性与用户数据保护。

29

2026.01.26

json数据格式
json数据格式

JSON是一种轻量级的数据交换格式。本专题为大家带来json数据格式相关文章,帮助大家解决问题。

418

2023.08.07

json是什么
json是什么

JSON是一种轻量级的数据交换格式,具有简洁、易读、跨平台和语言的特点,JSON数据是通过键值对的方式进行组织,其中键是字符串,值可以是字符串、数值、布尔值、数组、对象或者null,在Web开发、数据交换和配置文件等方面得到广泛应用。本专题为大家提供json相关的文章、下载、课程内容,供大家免费下载体验。

535

2023.08.23

jquery怎么操作json
jquery怎么操作json

操作的方法有:1、“$.parseJSON(jsonString)”2、“$.getJSON(url, data, success)”;3、“$.each(obj, callback)”;4、“$.ajax()”。更多jquery怎么操作json的详细内容,可以访问本专题下面的文章。

311

2023.10.13

go语言处理json数据方法
go语言处理json数据方法

本专题整合了go语言中处理json数据方法,阅读专题下面的文章了解更多详细内容。

77

2025.09.10

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

443

2023.08.02

lambda表达式
lambda表达式

Lambda表达式是一种匿名函数的简洁表示方式,它可以在需要函数作为参数的地方使用,并提供了一种更简洁、更灵活的编码方式,其语法为“lambda 参数列表: 表达式”,参数列表是函数的参数,可以包含一个或多个参数,用逗号分隔,表达式是函数的执行体,用于定义函数的具体操作。本专题为大家提供lambda表达式相关的文章、下载、课程内容,供大家免费下载体验。

207

2023.09.15

俄罗斯Yandex引擎入口
俄罗斯Yandex引擎入口

2026年俄罗斯Yandex搜索引擎最新入口汇总,涵盖免登录、多语言支持、无广告视频播放及本地化服务等核心功能。阅读专题下面的文章了解更多详细内容。

158

2026.01.28

热门下载

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

精品课程

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

共23课时 | 3万人学习

C# 教程
C# 教程

共94课时 | 7.8万人学习

Java 教程
Java 教程

共578课时 | 52.5万人学习

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

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