springboot3 声明式 HTTP 接口

1 介绍

        在 Spring 6 和 Spring Boot 3 中,我们可以使用 Java 接口来定义声明式的远程 HTTP 服务。这种方法受到 Feign 等流行 HTTP 客户端库的启发,与在 Spring Data 中定义 Repository 的方法类似。

        声明式 HTTP 接口包括用于 HTTP exchange 的注解方法。我们可以通过使用带注解的 Java 接口来简单地表达远程 API 的细节,然后让 Spring 生成实现该接口并执行 exchange 的代理。这有助于减少样板代码的编写。

1.1 Exchange 方法

   @HttpExchange 是我们可以应用于 HTTP 接口及其 exchange 方法的根注解。如果我们将其应用于接口层,那么它就会应用于所有 exchange 方法。这对于指定所有接口方法的共同属性(如 content type 或 URL 前缀)非常有用。所有 HTTP 方法都有对应的注解:

  • @GetExchange 用于 HTTP GET 请求。
  • @PostExchange 用于 HTTP POST 请求。
  • @PutExchange 用于 HTTP PUT 请求。
  • @PatchExchange 用于 HTTP PATCH 请求。
  • @DelectExchange 用于 HTTP DELETE 请求。

        让我们使用不同的 HTTP 方法注解,来为远程 API 定义一个声明式的 HTTP 接口:

interface BooksService {@GetExchange("/books")List<Book> getBooks();@GetExchange("/books/{id}")Book getBook(@PathVariable long id);@PostExchange("/books")Book saveBook(@RequestBody Book book);@DeleteExchange("/books/{id}")ResponseEntity<Void> deleteBook(@PathVariable long id);
}

        注意,所有 HTTP 方法注解都是用 @HttpExchange 元注解的。因此,@GetExchange("/books") 等同于 @HttpExchange(url = "/books",method = "GET")

1.2 方法参数

        在上述示例接口中,我们在方法参数中使用了 @PathVariable 和 @RequestBody 注解。此外,我们还可以为 exchange 方法使用以下参数、注解:

  • URI: 动态设置请求的 URL,覆盖注解属性。
  • HttpMethod:动态设置请求的 HTTP 方法,覆盖注解属性。
  • @RequestHeader: 添加请求头信息,参数可以是 Map 或 MultiValueMap
  • @PathVariable:替换请求 URL 中的占位符参数。
  • @RequestBody:提供的请求体可以是要序列化的对象,也可以是响应式流 publisher(如 Mono 或 Flux)。
  • @RequestParam:添加请求参数,参数可以是 Map 或 MultiValueMap
  • @CookieValue:添加 cookie,参数可以是 Map 或 MultiValueMap

        注意,只有 Content Type 为 application/x-www-form-urlencoded 的请求才会在请求体中对请求参数进行编码。否则,请求参数将作为 URL 查询参数添加。

1.3 返回值

        在我们的示例接口中,exchange 方法返回的是阻塞式的普通值。声明式 HTTP 接口 exchange 方法既支持阻塞式的返回值,也支持响应式返回值。此外,我们可以选择只返回特定的响应信息,如状态码或响应头。如果我们对服务响应完全不感兴趣,也可以返回 void

        HTTP 接口 exchange 方法支持以下返回值:

  • voidMono<Void>:执行请求并丢弃响应内容。
  • HttpHeadersMono<HttpHeaders>: 执行请求,丢弃响应体,返回响应头。
  • <T>Mono<T>:执行请求,并将响应体解码为所声明的类型。
  • <T>Flux<T>:执行请求,并将响应体解码为所声明类型的数据流。
  • ResponseEntity<Void>Mono<ResponseEntity<Void>>:执行请求,丢弃响应体,并返回一个包含状态和响应头的 ResponseEntity
  • ResponseEntity<T>Mono<ResponseEntity<T>>:执行请求,并返回一个包含状态、响应头和解码后的响应体 ResponseEntity
  • Mono<ResponseEntity<Flux<T>>:执行请求,并返回一个包含状态、响应头和解码后的响应体 ResponseEntity

        我们还可以使用 ReactiveAdapterRegistry 中注册的任何其他异步或响应式类型。

2 客户端代理实现

        既然我们已经定义了 HTTP 服务接口,就需要创建一个代理来实现该接口并执行 exchange。

2.1 Proxy Factory

        Spring 为我们提供了一个 HttpServiceProxyFactory,我们可以用它为 HTTP 接口生成一个客户端代理:

HttpServiceProxyFactory httpServiceProxyFactory = HttpServiceProxyFactory.builder(WebClientAdapter.forClient(webClient)).build();
booksService = httpServiceProxyFactory.createClient(BooksService.class);

        要使用提供的工厂创建代理,除了 HTTP 接口之外,我们还需要一个响应式 Web 客户端的实例:

WebClient webClient = WebClient.builder().baseUrl(serviceUrl).build();

        现在,我们可以将客户端代理实例注册为 Spring Bean 或组件,并用它请求 REST 服务。

2.2 异常处理

        默认情况下,WebClient 会对任何客户端或服务器错误 HTTP 状态代码抛出 WebClientResponseException。我们可以通过注册一个默认的 response status handler 来自定义异常处理,该 handler 适用于通过客户端执行的所有响应:

BooksClient booksClient = new BooksClient(WebClient.builder().defaultStatusHandler(HttpStatusCode::isError, resp ->Mono.just(new MyServiceException("Custom exception"))).baseUrl(serviceUrl).build());

        如此一来,如果我们请求的 book 不存在,我们就会收到一个自定义异常:

BooksService booksService = booksClient.getBooksService();
assertThrows(MyServiceException.class, () -> booksService.getBook(9));

2.3 项目实战基本配置

package com.ywz.framework.webClient;import com.ywz.common.SymmetricAlgorithmUtils;
import com.ywz.framework.property.FastAiProperties;
import com.ywz.framework.webClient.service.FastAiChatService;
import io.netty.channel.ChannelOption;
import io.netty.handler.timeout.ReadTimeoutHandler;
import io.netty.handler.timeout.WriteTimeoutHandler;
import lombok.AllArgsConstructor;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.http.client.reactive.ReactorClientHttpConnector;
import org.springframework.web.reactive.function.client.ExchangeStrategies;
import org.springframework.web.reactive.function.client.WebClient;
import org.springframework.web.reactive.function.client.support.WebClientAdapter;
import org.springframework.web.service.invoker.HttpServiceProxyFactory;
import reactor.netty.http.client.HttpClient;/*** 类描述 -> WebFlux webClient配置类** @Author: ywz* @Date: 2025/04/25*/
@Configuration
@AllArgsConstructor
public class WebClientConfig {private final FastAiProperties fastAiProperties;/*** 方法描述 -> 获取httpClient客户端** @Author: ywz* @Date: 2025/04/25*/@Beanpublic HttpClient httpClient() {return HttpClient.create().option(ChannelOption.CONNECT_TIMEOUT_MILLIS, 30000)  //连接超时.doOnConnected(conn -> {conn.addHandlerLast(new ReadTimeoutHandler(10)); //读超时conn.addHandlerLast(new WriteTimeoutHandler(10)); //写超时});}/*** 方法描述 -> 获取FastAi对话服务接口** @param httpClient httpClient客户端* @Author: ywz* @Date: 2025/04/25*/@Beanpublic FastAiChatService fastAiChatService(HttpClient httpClient) {// 构建WebClient实例,设置基础URL为解密后的地址WebClient client = WebClient.builder().baseUrl(SymmetricAlgorithmUtils.decrypt(fastAiProperties.baseUrl()))// 设置默认请求头,包括内容类型和授权信息.defaultHeaders(headers -> {headers.add("Content-Type", "application/json");headers.add("Authorization", "Bearer " + SymmetricAlgorithmUtils.decrypt(fastAiProperties.yiZhouKey()));})// 使用传入的httpClient进行请求.clientConnector(new ReactorClientHttpConnector(httpClient))// 配置交换策略,设置最大内存大小.exchangeStrategies(ExchangeStrategies.builder().codecs(clientCodecConfigurer ->clientCodecConfigurer.defaultCodecs().maxInMemorySize(1024 << 8)).build()).build();// 创建HttpServiceProxyFactory,用于生成服务接口的代理HttpServiceProxyFactory factory = HttpServiceProxyFactory.builderFor(WebClientAdapter.create(client)).build();// 返回FastAiChatService接口的实现return factory.createClient(FastAiChatService.class);}
}

3 测试

        让我们看看如何测试我们的示例中声明式 HTTP 接口,以及执行交互的客户端代理。

3.1 使用 Mockito

        由于我们的目标是测试使用声明式 HTTP 接口创建的客户端代理,因此需要使用 Mockito 的 deep stubbing 功能来模拟底层 WebClient 的 fluent API:

@Mock(answer = Answers.RETURNS_DEEP_STUBS)
private WebClient webClient;

        现在,我们可以使用 Mockito 的 BDD 方法链式调用 WebClient 方法,并提供模拟响应:

given(webClient.method(HttpMethod.GET).uri(anyString(), anyMap()).retrieve().bodyToMono(new ParameterizedTypeReference<List<Book>>(){})).willReturn(Mono.just(List.of(new Book(1,"Book_1", "Author_1", 1998),new Book(2, "Book_2", "Author_2", 1999))));

        模拟响应就绪后,我们就可以使用 HTTP 接口定义的方法调用我们的服务了:

BooksService booksService = booksClient.getBooksService();
Book book = booksService.getBook(1);
assertEquals("Book_1", book.title());

3.2 使用 MockServer

        如果我们不想模拟 WebClient,可以使用 MockServer 这样的库生成并返回固定的 HTTP 响应:

new MockServerClient(SERVER_ADDRESS, serverPort).when(request().withPath(PATH + "/1").withMethod(HttpMethod.GET.name()),exactly(1)).respond(response().withStatusCode(HttpStatus.SC_OK).withContentType(MediaType.APPLICATION_JSON).withBody("{\"id\":1,\"title\":\"Book_1\",\"author\":\"Author_1\",\"year\":1998}"));

        现在已经准备好了模拟的响应和正在运行的模拟服务器(mock server,),可以调用我们的服务了。

BooksClient booksClient = new BooksClient(WebClient.builder().baseUrl(serviceUrl).build());
BooksService booksService = booksClient.getBooksService();
Book book = booksService.getBook(1);
assertEquals("Book_1", book.title());

        此外,还可以验证我们的测试代码是否调用了正确的模拟服务。

mockServer.verify(HttpRequest.request().withMethod(HttpMethod.GET.name()).withPath(PATH + "/1"),VerificationTimes.exactly(1)
);

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

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

相关文章

多级缓存架构设计与实践经验

多级缓存架构设计与实践经验 在互联网大厂Java求职者的面试中&#xff0c;经常会被问到关于多级缓存的架构设计和实践经验。本文通过一个故事场景来展示这些问题的实际解决方案。 第一轮提问 面试官&#xff1a;马架构&#xff0c;欢迎来到我们公司的面试现场。请问您对多级…

Mac「brew」快速安装Redis

安装Redis 步骤 1&#xff1a;安装 Redis 打开终端&#xff08;Terminal&#xff09;。 运行以下命令安装 Redis&#xff1a; brew install redis步骤 2&#xff1a;启动 Redis 安装完成后&#xff0c;可以使用以下命令启动 Redis 服务&#xff1a; brew services start redis…

文献阅读(一)植物应对干旱的生理学反应 | The physiology of plant responses to drought

分享一篇Science上的综述文章&#xff0c;主要探讨了植物应对干旱的生理机制&#xff0c;强调通过调控激素信号提升植物耐旱性、保障粮食安全的重要性。 摘要 干旱每年致使农作物产量的损失&#xff0c;比所有病原体造成损失的总和还要多。为适应土壤中的湿度梯度变化&#x…

if consteval

if consteval 是 C23 引入的新特性&#xff0c;该特性是关于immediate function 的&#xff0c;即consteval function。用于在编译时检查当前是否处于 立即函数上下文&#xff08;即常量求值环境&#xff09;&#xff0c;并根据结果选择执行不同的代码路径。它是对 std::is_con…

MANIPTRANS:通过残差学习实现高效的灵巧双手操作迁移

25年3月来自北京通用 AI 国家重点实验室、清华大学和北大的论文“ManipTrans: Efficient Dexterous Bimanual Manipulation Transfer via Residual Learning”。 人手在交互中起着核心作用&#xff0c;推动着灵巧机器人操作研究的不断深入。数据驱动的具身智能算法需要精确、大…

Field访问对象int字段,对象访问int字段,通过openjdk17 C++源码看对象字段访问原理

在Java反射机制中&#xff0c;访问对象的int类型字段值&#xff08;如field.getInt(object)&#xff09;的底层实现涉及JVM对内存偏移量的计算与直接内存访问。本文通过分析OpenJDK 17源码&#xff0c;揭示这一过程的核心实现逻辑。 一、字段偏移量计算 1. Java层初始化偏移量…

Java查询数据库表信息导出Word

参考: POI生成Word多级标题格式_poi设置word标题-CSDN博客 1.概述 使用jdbc查询数据库把表信息导出为word文档, 导出为word时需要下载word模板文件。 已实现数据库: KingbaseES, 实现代码: 点击跳转 2.效果图 2.1.生成word内容 所有数据库合并 数据库不合并 2.2.生成文件…

Qt中的全局函数讲解集合(全)

在头文件<QtGlobal>中包含了Qt的全局函数&#xff0c;现在就这些全局函数一一详解。 1.qAbs 原型&#xff1a; template <typename T> T qAbs(const T &t)一个用于计算绝对值的函数。它可以用于计算各种数值类型的绝对值&#xff0c;包括整数、浮点数等 示…

AI与IT协同的典型案例

简介 本篇代码示例展示了IT从业者如何与AI协同工作&#xff0c;发挥各自优势。这些案例均来自2025年的最新企业实践&#xff0c;涵盖了不同IT岗位的应用场景。 一、GitHub Copilot生成代码框架 开发工程师AI协作示例&#xff1a;利用GitHub Copilot生成代码框架&#xff0c;…

三网通电玩城平台系统结构与源码工程详解(二):Node.js 服务端核心逻辑实现

本篇文章将聚焦服务端游戏逻辑实现&#xff0c;以 Node.js Socket.io 作为主要通信与逻辑处理框架&#xff0c;展开用户登录验证、房间分配、子游戏调度与事件广播机制的剖析&#xff0c;并附上多个核心代码段。 一、服务端文件结构概览 /server/├── index.js …

【prompt是什么?有哪些技巧?】

Prompt&#xff08;提示词&#xff09;是什么&#xff1f; Prompt 是用户输入给AI模型&#xff08;如ChatGPT、GPT-4等&#xff09;的指令或问题&#xff0c;用于引导模型生成符合预期的回答。它的质量直接影响AI的输出效果。 Prompt 的核心技巧 1. 明确目标&#xff08;Clar…

堆和二叉树--数据结构初阶(3)(C/C++)

文章目录 前言理论部分堆的模拟实现:(这里举的大根堆)堆的创建二叉树的遍历二叉树的一些其他功能实现 作业部分 前言 这期的话讲解的是堆和二叉树的理论部分和习题部分 理论部分 二叉树的几个性质:1.对于任意一个二叉树&#xff0c;度为0的节点比度为2的节点多一个 2.对于完全…

Dockerfile讲解与示例汇总

容器化技术已经成为应用开发和部署的标准方式,而Docker作为其中的佼佼者,以其轻量、高效、可移植的特性,深受开发者和运维人员的喜爱。本文将从实用角度出发,分享各类常用服务的Docker部署脚本与最佳实践,希望能帮助各位在容器化之路上少走弯路。 无论你是刚接触Docker的…

在QGraphicsView中精确地以鼠标为锚缩放图片

在pyqt中以鼠标所在位置为锚点缩放图片-CSDN博客中的第一个示例中&#xff0c;通过简单设置&#xff1a; self.setTransformationAnchor(QGraphicsView.AnchorUnderMouse) 使得QGraphicsView具有了以鼠标为锚进行缩放的功能。但是&#xff0c;其内部应当是利用了滚动条的移动来…

制造工厂如何借助电子看板实现高效生产管控

在当今高度竞争的制造业环境中&#xff0c;许多企业正面临着严峻的管理和生产挑战。首先&#xff0c;管理流程落后&#xff0c;大量工作仍依赖"人治"方式&#xff0c;高层管理者理论知识薄弱且不愿听取专业意见。其次&#xff0c;生产过程控制能力不足&#xff0c;导…

在 C# .NET 中驾驭 JSON:使用 Newtonsoft.Json 进行解析与 POST 请求实战

JSON (JavaScript Object Notation) 已经成为现代 Web 应用和服务之间数据交换的通用语言。无论你是开发后端 API、与第三方服务集成&#xff0c;还是处理配置文件&#xff0c;都绕不开 JSON 的解析与生成。在 C# .NET 世界里&#xff0c;处理 JSON 有多种选择&#xff0c;其中…

Debian10系统安装,磁盘分区和扩容

1、说明 过程记录信息有些不全&#xff0c;仅作为参考。如有其它疑问&#xff0c;欢迎留言。 2、ISO下载 地址&#xff1a;debian-10.13.0镜像地址 3、开始安装 3.1、选择图形界面 3.2、选择中文语言 3.3、选择中国区域 3.4、按照提示继续 3.5、选择一个网口 3.6、创建管…

1.10软考系统架构设计师:优秀架构设计师 - 练习题附答案及超详细解析

优秀架构设计师综合知识单选题 每道题均附有答案解析&#xff1a; 题目1 衡量优秀系统架构设计师的核心标准不包括以下哪项&#xff1f; A. 技术全面性与底层系统原理理解 B. 能够独立完成模块开发与调试 C. 与利益相关者的高效沟通与协调能力 D. 对业务需求和技术趋势的战略…

MPI Code for Ghost Data Exchange in 3D Domain Decomposition with Multi-GPUs

MPI Code for Ghost Data Exchange in 3D Domain Decomposition with Multi-GPUs Here’s a comprehensive MPI code that demonstrates ghost data exchange for a 3D domain decomposition across multiple GPUs. This implementation assumes you’re using CUDA-aware MPI…

计算机考研精炼 计网

第 19 章 计算机网络体系结构 19.1 基本概念 19.1.1 计算机网络概述 1.计算机网络的定义、组成与功能 计算机网络是一个将分散的、具有独立功能的计算机系统&#xff0c;通过通信设备与线路连接起来&#xff0c;由功能完善的软件实现资源共享和信息传递的系统。 …