0

0

使用RestTemplate获取并处理API数据:过滤、转换与定制化输出

聖光之護

聖光之護

发布时间:2025-09-28 12:00:05

|

707人浏览过

|

来源于php中文网

原创

使用resttemplate获取并处理api数据:过滤、转换与定制化输出

本文详细介绍了如何使用Spring RestTemplate从外部API获取数据,并利用Java Stream API对复杂数据结构进行高效的过滤和转换。教程涵盖了去重、属性重命名以及将原始数据映射为自定义API输出格式的完整流程,旨在帮助开发者构建更灵活、可控的数据消费与展示逻辑。

引言:API数据消费与处理的挑战

在现代应用开发中,与外部API交互是常见的需求。Spring框架的RestTemplate提供了一种便捷的方式来消费RESTful服务。然而,从外部API获取的数据往往是复杂的、嵌套的Java对象列表。在将这些数据展示给最终用户或提供给自己的API之前,我们通常需要对其进行一系列处理,例如过滤重复项、选择性地提取字段、重命名属性,甚至将数据转换成完全不同的结构。本教程将引导您完成这一过程,特别关注如何利用Java Stream API高效地实现这些数据操作。

初始API调用与数据模型

假设我们正在使用RestTemplate从一个天气API获取省份数据。初始设置包括一个RestTemplate调用和一个用于反序列化JSON响应的DTO(Data Transfer Object)结构。

1. RestTemplate 获取数据

首先,我们有一个静态方法用于通过RestTemplate调用外部API:

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;
}

这个方法会返回一个Provinces对象,其中包含了从API获取的原始省份数据。

2. Provinces 和 ProvincesData DTOs

为了正确地将JSON响应映射到Java对象,我们定义了两个DTO类:Provinces和ProvincesData。

Provinces类:

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

@JsonIgnoreProperties(ignoreUnknown = true)
public class Provinces {

    @JsonProperty("provincias") // 注意:原始问题中是"provincial",这里修正为更常见的"provincias"
    private List provinces;

    public Provinces() {}

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

    @JsonProperty("provincias")
    public List getProvinces() { // 修正方法名,遵循Java命名规范
        return provinces;
    }

    @JsonProperty("Test") // 原始问题中此处为"Test",可能为测试用途,实际应与JSON字段匹配或移除
    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") // 原始问题中此处为"Test",可能为测试用途,实际应与JSON字段匹配或移除
    public void setCodProv(String codProv) {
        this.codProv = codProv;
    }

    @JsonProperty("NOMBRE_PROVINCIA") // 补充JsonProperty注解以确保正确映射
    public String getNomeProvincia() {
        return nomeProvincia;
    }

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

    @JsonProperty("CODAUTON") // 补充JsonProperty注解
    public String getCodAuton() {
        return codAuton;
    }

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

    @JsonProperty("COMUNIDAD_CIUDAD_AUTONOMA") // 补充JsonProperty注解
    public String getComunidadeCidadeAutonoma() {
        return comunidadeCidadeAutonoma;
    }

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

注意:在上述DTO中,我修正了Provinces类中@JsonProperty的名称,以及方法命名以符合Java标准。ProvincesData中也补充了@JsonProperty注解,以确保字段的正确映射。原始问题中一些set方法上的@JsonProperty("Test")看起来是测试代码,在实际生产中应移除或修正。

数据过滤与转换的需求

原始问题提出,从API获取的数据是一个List,但需要对其进行以下处理:

  1. 去重:根据codAuton字段去除重复的省份数据。
  2. 属性重命名与精简:构建一个只包含所需数据的新列表,并更改属性名称,以便在自己的API中以更友好的标题展示。

解决方案:结合Java Stream API进行数据处理

Java 8引入的Stream API为集合操作提供了强大且富有表现力的方式,非常适合进行数据过滤、转换和聚合。

燕雀Logo
燕雀Logo

为用户提供LOGO免费设计在线生成服务

下载

1. 在数据获取层进行去重过滤

为了在数据返回之前就完成去重,我们可以在restTemplateProvince方法中直接应用Stream API。这里我们将根据codAuton字段进行去重。

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

public class Templates { // 假设这个类名为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);

        // 使用Stream API进行去重过滤
        List includedCodAutons = new ArrayList<>(); // 用于跟踪已经包含的codAuton
        List filteredProvinces = province.getProvinces()
            .stream()
            .filter(p -> {
                // 如果includedCodAutons中不包含当前p.getCodAuton(),则添加并返回true(保留)
                if (!includedCodAutons.contains(p.getCodAuton())) {
                    includedCodAutons.add(p.getCodAuton());
                    return true;
                }
                // 否则返回false(过滤掉)
                return false;
            })
            .collect(Collectors.toList());

        // 将过滤后的列表设置回Provinces对象
        province.setProvinces(filteredProvinces);
        return province;
    }
}

在这个修改后的restTemplateProvince方法中:

  • 我们首先像往常一样获取Provinces对象。
  • 然后,通过province.getProvinces().stream()创建一个数据流。
  • filter操作结合一个辅助includedCodAutons列表来判断codAuton是否已出现过。这种方法在处理少量数据时简洁易懂。
  • collect(Collectors.toList())将过滤后的元素收集回一个新的List
  • 最后,我们更新Provinces对象中的省份列表,并返回它。

2. 在业务逻辑层进行数据转换与重命名

尽管我们在数据获取层已经完成了去重,但如果您的API需要返回一个更精简、属性名更友好的列表,或者甚至是一个List,那么还需要进一步的转换。这通常在Service层完成。

首先,定义一个新的DTO来表示我们希望在API中展示的精简数据结构:

// CustomProvinceView.java
public class CustomProvinceView {
    private String autonomousCommunityCode; // 重命名codAuton
    private String autonomousCommunityName; // 重命名comunidadeCidadeAutonoma

    public CustomProvinceView(String autonomousCommunityCode, String autonomousCommunityName) {
        this.autonomousCommunityCode = autonomousCommunityCode;
        this.autonomousCommunityName = autonomousCommunityName;
    }

    // Getters
    public String getAutonomousCommunityCode() {
        return autonomousCommunityCode;
    }

    public String getAutonomousCommunityName() {
        return autonomousCommunityName;
    }

    // Setters (可选,如果不需要修改,可以省略)
    public void setAutonomousCommunityCode(String autonomousCommunityCode) {
        this.autonomousCommunityCode = autonomousCommunityCode;
    }

    public void setAutonomousCommunityName(String autonomousCommunityName) {
        this.autonomousCommunityName = autonomousCommunityName;
    }
}

接下来,修改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
    RestTemplate restTemplate; // 注入RestTemplate

    public List getAllProvincesFilteredAndTransformed() {
        // 调用Templates类中经过去重处理的方法
        Provinces listOfProvinces = Templates.restTemplateProvince(restTemplate);

        // 对去重后的ProvincesData列表进行转换
        return listOfProvinces.getProvinces()
            .stream()
            .map(p -> new CustomProvinceView(p.getCodAuton(), p.getComunidadeCidadeAutonoma()))
            .collect(Collectors.toList());
    }

    // 如果需要返回List,例如只包含社区名称:
    public List getAllAutonomousCommunityNames() {
        Provinces listOfProvinces = Templates.restTemplateProvince(restTemplate);
        return listOfProvinces.getProvinces()
            .stream()
            .map(ProvincesData::getComunidadeCidadeAutonoma)
            .collect(Collectors.toList());
    }
}

最后,更新您的RestController以返回转换后的数据:

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
    ProvinceService provinceService;

    @GetMapping("/provinces/custom")
    public List getCustomProvinces(){
        return provinceService.getAllProvincesFilteredAndTransformed();
    }

    @GetMapping("/provinces/community-names")
    public List getCommunityNames(){
        return provinceService.getAllAutonomousCommunityNames();
    }
}

通过以上步骤,我们实现了:

  1. 使用RestTemplate获取原始数据。
  2. 在数据获取阶段对数据进行去重处理。
  3. 在业务逻辑层将去重后的数据转换成符合API输出需求的新DTO列表,并重命名了属性。
  4. 提供了另一种将数据转换为List的示例。

注意事项与最佳实践

  • 性能考量:在filter操作中使用List.contains()对于非常大的列表(百万级别以上)可能会有性能问题,因为它是一个O(n)操作。对于大数据集,更高效的去重方法是使用Set:
    Set includedCodAutons = new HashSet<>();
    List filteredProvinces = province.getProvinces().stream()
        .filter(p -> includedCodAutons.add(p.getCodAuton())) // Set.add()返回true表示添加成功(不重复),false表示已存在(重复)
        .collect(Collectors.toList());

    这种方式将去重操作的平均时间复杂度降低到O(1),整体过滤效率更高。

  • DTO设计:为外部API响应和内部API输出设计不同的DTO是良好的实践。外部API DTO(如Provinces和ProvincesData)应尽可能匹配外部API的结构,以便于反序列化。而内部API DTO(如CustomProvinceView)则应根据您的业务需求和客户端展示需求进行优化,只包含必要字段且命名清晰。
  • 职责分离:将数据获取、过滤和转换逻辑分别放置在不同的层(如Templates类、Service层)有助于保持代码的模块化和可维护性。Templates负责原始数据获取和初步的通用处理,Service层负责业务逻辑和数据转换以满足特定API需求。
  • 错误处理:在实际应用中,RestTemplate调用应该包含适当的错误处理机制,例如try-catch块来捕获RestClientException,并处理网络问题或API返回的错误状态码

总结

本教程展示了如何利用Spring RestTemplate和Java Stream API有效地处理从外部API获取的复杂数据。通过分阶段的过滤和转换,我们能够将原始、可能冗余的数据转化为精简、结构清晰且符合特定API输出要求的格式。掌握这些技术将使您在构建健壮且灵活的Spring Boot

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

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

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

115

2025.08.06

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

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

30

2026.01.26

spring boot框架优点
spring boot框架优点

spring boot框架的优点有简化配置、快速开发、内嵌服务器、微服务支持、自动化测试和生态系统支持。本专题为大家提供spring boot相关的文章、下载、课程内容,供大家免费下载体验。

135

2023.09.05

spring框架有哪些
spring框架有哪些

spring框架有Spring Core、Spring MVC、Spring Data、Spring Security、Spring AOP和Spring Boot。详细介绍:1、Spring Core,通过将对象的创建和依赖关系的管理交给容器来实现,从而降低了组件之间的耦合度;2、Spring MVC,提供基于模型-视图-控制器的架构,用于开发灵活和可扩展的Web应用程序等。

390

2023.10.12

Java Spring Boot开发
Java Spring Boot开发

本专题围绕 Java 主流开发框架 Spring Boot 展开,系统讲解依赖注入、配置管理、数据访问、RESTful API、微服务架构与安全认证等核心知识,并通过电商平台、博客系统与企业管理系统等项目实战,帮助学员掌握使用 Spring Boot 快速开发高效、稳定的企业级应用。

70

2025.08.19

Java Spring Boot 4更新教程_Java Spring Boot 4有哪些新特性
Java Spring Boot 4更新教程_Java Spring Boot 4有哪些新特性

Spring Boot 是一个基于 Spring 框架的 Java 开发框架,它通过 约定优于配置的原则,大幅简化了 Spring 应用的初始搭建、配置和开发过程,让开发者可以快速构建独立的、生产级别的 Spring 应用,无需繁琐的样板配置,通常集成嵌入式服务器(如 Tomcat),提供“开箱即用”的体验,是构建微服务和 Web 应用的流行工具。

34

2025.12.22

Java Spring Boot 微服务实战
Java Spring Boot 微服务实战

本专题深入讲解 Java Spring Boot 在微服务架构中的应用,内容涵盖服务注册与发现、REST API开发、配置中心、负载均衡、熔断与限流、日志与监控。通过实际项目案例(如电商订单系统),帮助开发者掌握 从单体应用迁移到高可用微服务系统的完整流程与实战能力。

156

2025.12.24

PHP API接口开发与RESTful实践
PHP API接口开发与RESTful实践

本专题聚焦 PHP在API接口开发中的应用,系统讲解 RESTful 架构设计原则、路由处理、请求参数解析、JSON数据返回、身份验证(Token/JWT)、跨域处理以及接口调试与异常处理。通过实战案例(如用户管理系统、商品信息接口服务),帮助开发者掌握 PHP构建高效、可维护的RESTful API服务能力。

160

2025.11.26

clawdbot ai使用教程 保姆级clawdbot部署安装手册
clawdbot ai使用教程 保姆级clawdbot部署安装手册

Clawdbot是一个“有灵魂”的AI助手,可以帮用户清空收件箱、发送电子邮件、管理日历、办理航班值机等等,并且可以接入用户常用的任何聊天APP,所有的操作均可通过WhatsApp、Telegram等平台完成,用户只需通过对话,就能操控设备自动执行各类任务。

8

2026.01.29

热门下载

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

精品课程

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

共23课时 | 3万人学习

C# 教程
C# 教程

共94课时 | 7.9万人学习

Java 教程
Java 教程

共578课时 | 52.9万人学习

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

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