OpenFeign-查询参数-日期格式化:LocalDate、Date、@DateTimeFormat(低版本无效)

news2025/1/22 19:03:20

创建时间:2024-10-08

本文适用的依赖版本:
spring-boot-starter-parent:3.3.3
spring-cloud-starter-openfeign:4.1.3

一、场景

REST API 的查询接口中,日期查询参数 的格式一般是标准(ISO 8601)的日期字符串,比如:2024-10-04。常见的查询参数一般为:开始日期(beginDate),结束日期(endDate)。

数据类中日期类型一般使用 LocalDate 或者 Date,但是远程接口中接收的是字符串类型的日期。在使用OpenFeign调用远程接口时,FeignClient中的查询接口需要传递日期类型的查询参数,此时日期参数的参数类型,应该如何处理?

其中一种方法时,直接将FeignClient中的日期类型定为 String,然后在调用FeignClient接口的位置,将 LocalDate 或 Date 转为日期字符串。不过这种方法需要每次调用接口前都要做一次转换,比较繁琐,不够简洁,不推荐使用。

另一种是在FeignClient中的接口,使用 LocalDate 或 Date 类型表示日期,然后在实际调用远程接口时,由Feign框架将LocalDate或Date类型转为字符串类型的日期,此时需要使用@DateTimeFormat注解进行日期格式化。下面主要介绍这种方法。

二、@DateTimeFormat格式化日期

FeignClient

使用 @DateTimeFormat(iso = DateTimeFormat.ISO.DATE) 修饰日期查询参数,可以将日期类型转为字符串类型的日期。

@DateTimeFormat(iso = DateTimeFormat.ISO.DATE) ,代表指定的日期格式是 ISO 的日期格式。使用 @DateTimeFormat(pattern = “yyyy-MM-dd”),还可以指定其他的日期格式。

package com.example.hello_feign_client.feign.client;

import com.example.hello_common.model.query.LocalDateQuery;
import org.springframework.cloud.openfeign.FeignClient;
import org.springframework.cloud.openfeign.SpringQueryMap;
import org.springframework.format.annotation.DateTimeFormat;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestParam;

import java.time.LocalDate;
import java.util.Date;

@FeignClient(contextId = "dateFeignClient",
        name = "hello-feign-server",
        url = "${service.hello-feign-server.url}",
        path = "/date")
public interface DateFeignClient {

    @GetMapping("/localDate")
    String getDate(@RequestParam @DateTimeFormat(iso = DateTimeFormat.ISO.DATE) LocalDate localDate);

    @GetMapping("/localDateQuery")
    String getDateQuery(@SpringQueryMap LocalDateQuery localDateQuery);

    @GetMapping("/date")
    String getDate(@RequestParam @DateTimeFormat(iso = DateTimeFormat.ISO.DATE) Date date);
}

调用FeignClient

package com.example.hello_feign_client.web.date.controller;

import com.example.hello_common.model.query.LocalDateQuery;
import com.example.hello_feign_client.feign.client.DateFeignClient;
import lombok.RequiredArgsConstructor;
import org.springframework.format.annotation.DateTimeFormat;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;

import java.time.LocalDate;
import java.util.Date;

@RestController
@RequestMapping("/date")
@RequiredArgsConstructor
public class DateController {

    private final DateFeignClient dateFeignClient;

    @GetMapping("/localDate")
    public String getDate(@DateTimeFormat(iso = DateTimeFormat.ISO.DATE) LocalDate localDate) {
        return dateFeignClient.getDate(localDate);
    }

    @GetMapping("/localDateQuery")
    public String getDateQuery(LocalDateQuery localDateQuery) {
        return dateFeignClient.getDateQuery(localDateQuery);
    }

    @GetMapping("/date")
    public String getDate(@DateTimeFormat(iso = DateTimeFormat.ISO.DATE) Date date) {
        return dateFeignClient.getDate(date);
    }

}

LocalDateQuery

package com.example.hello_common.model.query;

import lombok.Data;
import org.springframework.format.annotation.DateTimeFormat;

import java.time.LocalDate;

@Data
public class LocalDateQuery {

    @DateTimeFormat(iso = DateTimeFormat.ISO.DATE)
    private LocalDate beginDate;

    @DateTimeFormat(iso = DateTimeFormat.ISO.DATE)
    private LocalDate endDate;

}

远程接口

package com.example.hello_feign_server.web.date.controller;

import com.example.hello_common.model.query.LocalDateQuery;
import org.springframework.format.annotation.DateTimeFormat;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;

import java.time.LocalDate;
import java.util.Date;

@RestController
@RequestMapping("/date")
public class DateController {

    @GetMapping("/localDate")
    public String getDate(@DateTimeFormat(iso = DateTimeFormat.ISO.DATE) LocalDate localDate) {
        return "根据日期(LocalDate)查询数据。查询参数:" + localDate;
    }

    @GetMapping("/localDateQuery")
    public String getDateQuery(LocalDateQuery localDateQuery) {
        return "根据日期Query(LocalDateQuery)查询数据。查询参数:" + localDateQuery;
    }

    @GetMapping("/date")
    public String getDate(@DateTimeFormat(iso = DateTimeFormat.ISO.DATE) Date date) {
        return "根据日期(Date)查询数据。查询参数:" + date;
    }

}

三、接口调用效果

远程接口

在这里插入图片描述
在这里插入图片描述

FeignClient调用远程接口效果

@RequestParam 直接传递请求参数

在这里插入图片描述

FeignClient接口调用日志:

在这里插入图片描述

@SpringQueryMap传递请求参数对象

在这里插入图片描述

FeignClient接口调用日志:

在这里插入图片描述

四、Date与@DateTimeFormat

@RequestParam + @DateTimeFormat + Date,有效

FeignClient 和 调用FeignClient 的代码,在上面的代码中已经给出。

调用效果:

在这里插入图片描述

FeignClient日志:

在这里插入图片描述

@SpringQueryMap + @DateTimeFormat + Date,无效

注意,在 @SpringQueryMap 修饰的参数对象中,如果使用了 java.util.Date 类型的字段,此时 @DateTimeFormat 是无效的。

比如下面代码中的 endDate

package com.example.hello_common.model.query;

import lombok.Data;
import org.springframework.format.annotation.DateTimeFormat;

import java.time.LocalDate;
import java.util.Date;

@Data
public class LocalDateQuery {

    @DateTimeFormat(iso = DateTimeFormat.ISO.DATE)
    private LocalDate beginDate;

    @DateTimeFormat(iso = DateTimeFormat.ISO.DATE)
    private Date endDate;

}

报错如下:

在这里插入图片描述
在这里插入图片描述

五、@DateTimeFormat在Feign低版本无效

经过测试,在 SpringBoot1.5 中,@RequestParam + @DateTimeFormat + Date是无效的。

请参考:

https://blog.csdn.net/hysxchina/article/details/102907861

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2196078.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

BiGRU-Transformer时间序列预测(多输入单预测)——基于Pytorch框架

1 介绍 本文将介绍一种基于Transformer和BiGRU(双向门控循环单元)的混合模型及其在时间序列预测中的应用。本模特适用于多输入单输出预测,适合风电预测,功率预测,负荷预测等等。 2 方法 首先,从Excel文件…

常见数据库等保测评检查命令汇总

MySQL数据库 select user, host FROM mysql.user;查看管理用户 show variables like validate%; 查看口令策略 show variables like %password%; 查看口令策略 show variables like %general%; 查看日志是否开启 show variables like general_log%; 可以看查询日…

电磁兼容(EMC):PCB设计里的镜像面究竟是个啥?

目录 1. 镜像面概念 2. 镜像面示意 3. 镜像面工作原理 4. 总结 PCB设计中涉及到高频信号处理时经常会听到一个叫做镜像面的概念。镜像面究竟是哪个面?和我们平时所说的地平面和电源平面有什么区别? 1. 镜像面概念 镜像面是指高频信号在完整平面上的…

退货单不能反价值重估的操作

财务的工作效率几何级式倍增。29号做的退货单,30号就做了发票。业务员工作质量却是堪忧,退货单做了2次。这样的问题是常态,该如何避免呢? 从提示文字分析,不能弃审, 一般是单据被下游单据锁定导致的&#…

算法 | 模拟

目录 替换所有的问号 题解: 提莫攻击 题解: Z字形变换 题解: 外观数列 题解: 数青蛙 题解: 替换所有的问号 1576. 替换所有的问号 - 力扣(LeetCode)https://leetcode.cn/problems/re…

AFSim仿真系统 --- 系统简解_03( Warlock模块 - 人在环路 在仿真领域中指的是AFSIM的操作员互动可视化应用程序)

我们就用保留单词 Warlock Warlock(在仿真领域中指的是AFSIM的操作员互动可视化应用程序--人在环路 ) Warlock是AFSIM的操作员环环相扣(Operator-in-the-Loop)视觉应用程序。它提供了一个图形环境,用于在运行时查看和…

【Verilog学习日常】—牛客网刷题—Verilog进阶挑战—VL45

异步FIFO 描述 请根据题目中给出的双口RAM代码和接口描述,实现异步FIFO,要求FIFO位宽和深度参数化可配置。 电路的接口如下图所示。 双口RAM端口说明: 端口名 I/O 描述 wclk input 写数据时钟 wenc input 写使能 waddr input 写…

Qt程序国际化

目录 程序的国际化 第一步:生成ts文件 第二步:生成并加载qm文件 程序的国际化 第一步:生成ts文件 Ts文件格式上实际是xml,记录界面的字符和代码中tr包含的字符的具体位置(文件位置,行号列号等&#xff0…

QT 实现图片查看工具

QT 实现图片查看工具 1、选择图像文件 单文件选择 QFileDialog::getOpenFileName多文件选择 QFileDialog::getOpenFileNamesQList<QString> imageNames = QFileDialog::getOpenFileNames(this,tr("打开图片"),"",tr("图片文件 (*.png *.jpg *.b…

【STM32 Blue Pill编程实例】-OLED显示DHT22传感器数据

OLED显示DHT22传感器数据 文章目录 OLED显示DHT22传感器数据1、DHT22介绍2、硬件准备与接线3、模块配置3.1 定时器配置3.2 DHT22引脚配置3.3 OLED配置4、代码实现在本文中,我们将介绍如何将 DHT22 温度和湿度传感器与 STM32 Blue Pill 开发板连接,并使用 HAL 库在 STM32CubeI…

车载音频焦点(二)

目录 1 可延迟的音频焦点 2 多音频区焦点管理 3 HAL 音频焦点 4 OEM 车载音频焦点服务 1 可延迟的音频焦点 在 Android 11 中,AAOS 开始支持请求获得可延迟的音频焦点。 当非瞬态焦点请求 与 当前焦点持有者交互 遭到拒绝时,前者可以延迟。 一旦焦点的变化导致延迟的请…

各省份自然灾害损失造成的直接经济损失数据(2009-2022年)

自然灾害是自然演变过程中不可避免的现象&#xff0c;它们对人类社会构成了巨大的威胁。中国作为一个自然灾害频发的国家&#xff0c;面临着种类繁多的灾害挑战&#xff0c;包括气象灾害、地质灾害、海洋灾害、生物灾害和森林草原火灾等。 数据来源&#xff1a;《中国环境统计…

数据分布过于集中 怎么办,python 人工智能 ,数据分析,机器学习pytorch tensorflow ,

数据分布过于集中&#xff0c;意味着数据的大部分值都聚集在某个特定区间内&#xff0c;这可能会导致统计分析的结果不够稳健&#xff0c;或者模型训练时出现过拟合等问题。针对这种情况&#xff0c;可以考虑以下几种方法来处理&#xff1a; 变换成 1. **数据转换**&#xff1…

笔记-stm32移植ucos

文章目录 一、UCOS的基础知识1.1 前后台系统:1.2 RTOS系统可剥夺型内核:前后台系统和RTOS系统 1.3 UCOS系统简介学习方法 二、ucossii移植Step1&#xff1a;在工程中建立存放UCOSS代码的文件夹UCOSIIStep2:向CORE文件夹添加文件Step3:向Config文件夹添加文件Step4:向port文件夹…

Knife4j-SpringBoot3-OpenAPI3:基本使用、生产环境关闭接口文档、配置文件、配置接口文档描述信息、OpenAPI3注解

版本&#xff1a; SpringBoot&#xff1a;3.3.3 Knife4j&#xff1a;4.5.0 创建时间&#xff1a;2024-10-08 一、官网 Knife4j 的 SpringBoot3 官方说明文档&#xff1a; https://doc.xiaominfo.com/docs/quick-start#spring-boot-3 springdoc官网&#xff1a;https://spring…

笔试强训day33

跳台阶扩展问题 一只青蛙一次可以跳上1级台阶&#xff0c;也可以跳上2级……它也可以跳上n级。求该青蛙跳上一个n级的台阶(n为正整数)总共有多少种跳法。 数据范围&#xff1a;1≤n≤201≤n≤20 进阶&#xff1a;空间复杂度 O(1)O(1) &#xff0c; 时间复杂度 O(1)O(1) 输入…

【EXCEL数据处理】000017 案例 保姆级教程,附多个操作案例。EXCEL Match和Index函数。

前言&#xff1a;哈喽&#xff0c;大家好&#xff0c;今天给大家分享一篇文章&#xff01;创作不易&#xff0c;如果能帮助到大家或者给大家一些灵感和启发&#xff0c;欢迎收藏关注哦 &#x1f495; 目录 【EXCEL数据处理】000017 案例 保姆级教程&#xff0c;附多个操作案例。…

MYSQL 事物隔离级别的区别与现象

事物的ACID属性本章不再赘述&#xff0c;本章主要描述事物的隔离级别及隔离级别导致的现象&#xff0c;日常工作中该如何选择MYSQL的隔离级别。 MYSQL事物的隔离级别及各隔离级别存在的问题如下&#xff1a; 隔离级别/问题脏读不可重复读幻读读未提交(Read-Uncommitted)✅✅✅…

JS 入门

文章目录 JS 入门一、JS 概述1、JS 特点2、JS 组成3、JS 初体验4、HTML引入JS 二、JS 基础语法1、变量声明2、基本数据类型3、引用数据类型1&#xff09;数组2&#xff09;对象3&#xff09;函数4&#xff09;null 4、运算符5、条件判断6、循环语句 三、JS 函数0、JS 函数特点1…