springboot学习第11期 - @HttpExchange

@HttpExchange 是什么

@HttpExchange 是springboot3提供的声明式发送HTTP请求客户端工具。

使用

第一步:根据第三方服务的接口声明

假如有第三方接口api:http://localhost:8080/api/hello/{id}?name={name},返回String类型。

按照上面的格式,创建第三方Client接口:

java 复制代码
@HttpExchange("/api/hello")
public interface ThirdClient {

    @GetExchange("/{id}")
    String sayHello(@PathVariable("id") int id, 
                    @RequestParam(value = "name", defaultValue = "World") String name
    );

    // 也可以返回 ResponseEntity, 如果想要获取响应其他信息的话
    // @GetExchange("/{id}")
    // ResponseEntity<String> sayHello(@PathVariable("id") int id, 
    //                 @RequestParam(value = "name", defaultValue = "World") String name
    // );

}

第二步:注入实现类

java 复制代码
@Configuration
public class HttpExchangeConfig {

    @Bean
    public ThirdClient thirdClient(RestClient.Builder builder) {
        RestClient restClient = builder
                .baseUrl("http://localhost:8080")
                .build();

        HttpServiceProxyFactory factory = HttpServiceProxyFactory
                .builderFor(RestClientAdapter.create(restClient))
                .build();

        return factory.createClient(ThirdClient.class);
    }
}

第三步:调用

java 复制代码
@Autowired
private ThirdClient thirdClient;


String s = thirdClient.sayHello(1, "aaa");

异常处理

本质上还是引用了RestClient发送请求,所以异常还是和RestClient异常一样。

通用异常是 RestClientException

4xx异常:HttpClientErrorException

5xx异常:HttpServerErrorException

网络错误:ResourceAccessException

重试

可以结合 spring retry 注解达到重试的效果

java 复制代码
@HttpExchange("/api/hello")
public interface ThirdClient {

    @GetExchange("/{id}")
    @Retryable(retryFor = {RestClientException.class}, noRetryFor = {HttpClientErrorException.class}, maxAttempts = 3, backoff = @Backoff(delay = 5000), listeners = {"customRetryListener"})
    ResponseEntity<String> sayHello(@PathVariable("id") int id, @RequestParam(value = "name", defaultValue = "World") String name);

}

这里还注入了一个自定义重试监听器,当执行重试操作的时候,可以打印一些自定义日志。

java 复制代码
@Component
public class CustomRetryListener implements RetryListener {
    private static final Logger log = LoggerFactory.getLogger(CustomRetryListener.class);

    @Override
    public <T, E extends Throwable> boolean open(RetryContext context, RetryCallback<T, E> callback) {
        // 第一次调用前触发
        log.info("开始重试操作 - 方法: {}", context.getAttribute(RetryContext.NAME));
        return true;
    }

    @Override
    public <T, E extends Throwable> void close(RetryContext context, RetryCallback<T, E> callback, Throwable throwable) {
        // 最后一次重试结束后触发(无论成功失败)
        if (throwable != null) {
            log.error("重试结束,最终失败 - 方法: {}, 异常: {}", context.getAttribute(RetryContext.NAME), throwable.toString());
        } else {
            log.info("重试结束,成功 - 方法: {}, 重试次数: {}", context.getAttribute(RetryContext.NAME), context.getRetryCount());
        }
    }

    @Override
    public <T, E extends Throwable> void onError(RetryContext context, RetryCallback<T, E> callback, Throwable throwable) {
        // 每次重试失败后触发
        log.warn("重试失败 - 方法: {}, 第 {} 次重试, 异常: {}",
                context.getAttribute(RetryContext.NAME),
                context.getRetryCount(),
                throwable.toString());
    }
}
相关推荐
shaoming37764 小时前
检查系统硬件配置是否满足PyCharm最低要求
android·spring boot·mysql
ffqws_4 小时前
Spring Boot 接收前端请求的四种参数方式
前端·spring boot·后端
傻瓜搬砖人5 小时前
SpringBoot整合Junit-Redis-打包
spring boot·redis·junit
Arya_aa7 小时前
数据字典模块–MapStruct对象转换与加密处理
spring boot
RuoyiOffice8 小时前
2026 年开源 BPM/工作流引擎大盘点:Flowable vs Camunda vs Activiti vs Turbo——谁才是企业级首选?
java·spring boot·后端·开源·流程图·ruoyi·anti-design-vue
贫民窟的勇敢爷们8 小时前
Spring Boot+Vue电商系统开发实战:架构设计与核心实现
vue.js·spring boot·后端
星光开发者9 小时前
基于springboot电动汽车租赁管理系统-计算机毕设 附源码 11217
javascript·spring boot·mysql·django·php·html5·express
苍煜9 小时前
SpringBoot Spring事务完整版详解:@Transactional注解实操 + 七大事务传播机制用法
spring boot·spring·oracle
ffqws_9 小时前
Spring Boot 配置读取全解析:从 application.yml 到 Java 对象的完整链路
java·数据库·spring boot
RuoyiOffice9 小时前
SpringBoot+Vue3 实现 OA 公文外来文与归档台账:外部收文、BPM办理、三类公文统一归档
spring boot·微服务·uni-app·vue·ruoyi·anti-design-vue·ruoyioffice