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());
    }
}
相关推荐
小马爱打代码2 小时前
Spring Boot:模块化实战 - 保持清晰架构
java·spring boot·架构
8***Z893 小时前
springboot 异步操作
java·spring boot·mybatis
i***13243 小时前
Spring BOOT 启动参数
java·spring boot·后端
WZTTMoon4 小时前
Spring Boot 启动全解析:4 大关键动作 + 底层逻辑
java·spring boot·后端
皮皮林5515 小时前
SpringBoot + nmap4j 获取端口信息
spring boot
毕设源码-钟学长5 小时前
【开题答辩全过程】以 基于springboot和协同过滤算法的线上点餐系统为例,包含答辩的问题和答案
java·spring boot·后端
h***04776 小时前
SpringBoot集成Flink-CDC,实现对数据库数据的监听
数据库·spring boot·flink
o***74176 小时前
Springboot中SLF4J详解
java·spring boot·后端
d***95628 小时前
springboot接入deepseek深度求索 java
java·spring boot·后端
z***02608 小时前
SpringBoot连接多数据源MySQL、SqlServer等(MyBatisPlus测试)
spring boot·mysql·sqlserver