扔掉HttpUtil!看看人家的HTTP客户端工具,那叫一个优雅!

我们平时开发项目时,就算是单体应用,也免不了要调用一下外部服务。此时就会用到HTTP客户端工具。之前使用过Hutool中的HttpUtil,虽然容易上手,但用起来颇为麻烦!今天给大家分享一款更好用的HTTP客户端工具Retrofit,你只需声明接口就可发起请求,无需连接、结果解析之类的重复操作,用起来够优雅!

简介

Retrofit是适用于AndroidJava且类型安全的HTTP客户端工具,在Github上已经有43k+Star。其最大的特性的是支持通过接口的方式发起HTTP请求,类似于我们用Feign调用微服务接口的那种方式。

Spring Boot是使用最广泛的Java开发框架,但是Retrofit官方并没有提供专门的Starter。于是有位老哥就开发了retrofit-spring-boot-starter,它实现了Retrofit与Spring Boot框架的快速整合,并且支持了诸多功能增强,极大简化开发。今天我们将使用这个第三方Starter来操作Retrofit。

使用

在Spring Boot中使用Retrofit是非常简单的,下面我们就来体验下。

依赖集成

有了第三方Starter的支持,集成Retrofit仅需一步,添加如下依赖即可。

xml 复制代码
<!--Retrofit相关依赖-->
<dependency>
  <groupId>com.github.lianjiatech</groupId>
  <artifactId>retrofit-spring-boot-starter</artifactId>
  <version>${retrofit-start.version}</version>
</dependency>

基本使用

我们将以调用mall电商实战项目中的接口为例,来讲解Retrofit的基本使用。这里简单介绍下mall项目,mall项目是一套基于 SpringBoot3 + Vue 的电商系统(Github标星60K),后端支持多模块和微服务架构,采用Docker和K8S部署。包括前台商城项目和后台管理系统,能支持完整的订单流程!涵盖商品、订单、购物车、权限、优惠券、会员、支付等功能!

项目演示:

  • 首先我们把mall项目的后台服务mall-admin给运行起来,这里我们会调用其中的登录接口和商品品牌管理接口,可以打开API文档查看下:http://localhost:8080/swagger-ui/index.html
  • 我们先来调用下登录接口,在项目的application.yml配置文件中添加mall-admin的服务地址;
yaml 复制代码
remote:
  baseUrl: http://localhost:8080/
  • 再通过@RetrofitClient声明一个Retrofit客户端,由于登录接口是通过POST形式,在body中传入JSON请求参数来调用的,这里使用到了@POST@Body注解;
java 复制代码
/**
 * @auther macrozheng
 * @description 定义Http接口,用于调用远程的UmsAdmin服务
 * @date 2025/6/24
 * @github https://github.com/macrozheng
 */
@RetrofitClient(baseUrl = "${remote.baseUrl}")
public interface UmsAdminApi {

  @POST("admin/login")
  CommonResult<LoginResult> login(@Body LoginParam loginParam);
}
  • 如果你不太明白这些注解是干嘛的,看下下面的表基本就懂了,更具体的话可以参考Retrofit官方文档;
  • 接下来在Controller中注入UmsAdminApi,然后进行调用;
java 复制代码
/**
 * @auther macrozheng
 * @description Retrofit远程调用测试接口
 * @date 2025/6/24
 * @github https://github.com/macrozheng
 */
@Tag(name = "RetrofitController", description = "Retrofit远程调用测试接口")
@RestController
@RequestMapping("/retrofit")
public class RetrofitController {

  @Autowired
  private UmsAdminApi umsAdminApi;
  @Autowired
  private TokenHolder tokenHolder;

  @Operation(summary = "调用远程登录接口获取token")
  @PostMapping(value = "/admin/login")
  public CommonResult<LoginResult> login(@RequestParam String username, @RequestParam String password) {
    CommonResult<LoginResult> result = umsAdminApi.login(new LoginParam(username, password));
    LoginResult loginResult = result.getData();
    if (result.getData() != null) {
      tokenHolder.putToken(loginResult.getTokenHead() + " " + loginResult.getToken());
    }
    return result;
  }
}  
  • 为方便后续调用需要登录认证的接口,我创建了TokenHolder这个类,把token存储到了Session中;
java 复制代码
/**
 * @auther macrozheng
 * @description 登录token存储(在Session中)
 * @date 2025/6/24
 * @github https://github.com/macrozheng
 */
@Component
public class TokenHolder {
  /**
   * 添加token
   */
  public void putToken(String token) {
    RequestAttributes ra = RequestContextHolder.getRequestAttributes();
    HttpServletRequest request = ((ServletRequestAttributes) ra).getRequest();
    request.getSession().setAttribute("token", token);
  }

  /**
   * 获取token
   */
  public String getToken() {
    RequestAttributes ra = RequestContextHolder.getRequestAttributes();
    HttpServletRequest request = ((ServletRequestAttributes) ra).getRequest();
    Object token = request.getSession().getAttribute("token");
    if(token!=null){
      return (String) token;
    }
    return null;
  }

}

注解式拦截器

由于商品品牌管理接口,需要添加登录认证头才可以正常访问,我们可以使用Retrofit中的注解式拦截器来实现。

  • 首先创建一个注解式拦截器TokenInterceptor继承BasePathMatchInterceptor,然后在doIntercept方法中给请求添加Authorization头;
java 复制代码
/**
 * @auther macrozheng
 * @description 给请求添加登录Token头的拦截器
 * @date 2025/6/24
 * @github https://github.com/macrozheng
 */
@Component
public class TokenInterceptor extends BasePathMatchInterceptor {
  @Autowired
  private TokenHolder tokenHolder;

  @Override
  protected Response doIntercept(Chain chain) throws IOException {
    Request request = chain.request();
    if (tokenHolder.getToken() != null) {
      request = request.newBuilder()
              .header("Authorization", tokenHolder.getToken())
              .build();
    }
    return chain.proceed(request);
  }
}
  • 创建调用品牌管理接口的客户端PmsBrandApi,使用@Intercept注解配置拦截器和拦截路径;
java 复制代码
/**
 * @auther macrozheng
 * @description 定义Http接口,用于调用远程的PmsBrand服务
 * @date 2025/6/24
 * @github https://github.com/macrozheng
 */
@RetrofitClient(baseUrl = "${remote.baseUrl}")
@Intercept(handler = TokenInterceptor.class, include = "/brand/**")
public interface PmsBrandApi {
  @GET("brand/list")
  CommonResult<CommonPage<PmsBrand>> list(@Query("pageNum") Integer pageNum, @Query("pageSize") Integer pageSize);

  @GET("brand/{id}")
  CommonResult<PmsBrand> detail(@Path("id") Long id);

  @POST("brand/create")
  CommonResult create(@Body PmsBrand pmsBrand);

  @POST("brand/update/{id}")
  CommonResult update(@Path("id") Long id, @Body PmsBrand pmsBrand);

  @GET("brand/delete/{id}")
  CommonResult delete(@Path("id") Long id);
}
  • 再在Controller中注入PmsBrandApi实例,并添加方法调用远程服务即可;
java 复制代码
/**
 * @auther macrozheng
 * @description Retrofit远程调用测试接口
 * @date 2025/6/24
 * @github https://github.com/macrozheng
 */
@Tag(name = "RetrofitController", description = "Retrofit远程调用测试接口")
@RestController
@RequestMapping("/retrofit")
public class RetrofitController {

  @Autowired
  private UmsAdminApi umsAdminApi;
  
  @Operation(summary = "调用远程接口分页查询品牌列表")
  @GetMapping(value = "/brand/list")
  public CommonResult<CommonPage<PmsBrand>> listBrand(@RequestParam(value = "pageNum", defaultValue = "1") Integer pageNum,
                                                      @RequestParam(value = "pageSize", defaultValue = "3") Integer pageSize) {
    return pmsBrandApi.list(pageNum, pageSize);
  }

  @Operation(summary = "调用远程接口获取指定id的品牌详情")
  @GetMapping(value = "/brand/{id}")
  public CommonResult<PmsBrand> brand(@PathVariable("id") Long id) {
    return pmsBrandApi.detail(id);
  }

  @Operation(summary = "调用远程接口添加品牌")
  @PostMapping(value = "/brand/create")
  public CommonResult createBrand(@RequestBody PmsBrand pmsBrand) {
    return pmsBrandApi.create(pmsBrand);
  }

  @Operation(summary = "调用远程接口更新指定id品牌信息")
  @PostMapping(value = "/brand/update/{id}")
  public CommonResult updateBrand(@PathVariable("id") Long id, @RequestBody PmsBrand pmsBrand) {
    return pmsBrandApi.update(id,pmsBrand);
  }

  @Operation(summary = "调用远程接口删除指定id的品牌")
  @GetMapping(value = "/delete/{id}")
  public CommonResult deleteBrand(@PathVariable("id") Long id) {
    return  pmsBrandApi.delete(id);
  }
}
  • 在Swagger中调用接口进行测试,发现已经可以成功调用。

全局拦截器

如果你想给所有请求都加个请求头的话,可以使用全局拦截器。

创建SourceInterceptor类继承GlobalInterceptor接口,然后在Header中添加source请求头。

java 复制代码
/**
 * @auther macrozheng
 * @description 全局拦截器,给请求添加source头
 * @date 2025/6/24
 * @github https://github.com/macrozheng
 */
@Component
public class SourceInterceptor implements GlobalInterceptor {

  @Override
  public Response intercept(Chain chain) throws IOException {
    Request request = chain.request();
    Request newReq = request.newBuilder()
            .addHeader("source", "retrofit")
            .build();
    return chain.proceed(newReq);
  }
}

配置

Retrofit的配置很多,下面我们讲讲日志打印、全局超时时间和全局请求重试这三种最常用的配置。

日志打印

  • 默认配置下Retrofit使用basic日志策略,打印的日志非常简单;
  • 我们可以将application.yml中的retrofit.global-log.log-strategy属性修改为body来打印最全日志;
yaml 复制代码
retrofit:
  # 全局日志配置
  global-log:
    # 启用日志打印
    enable: true
    # 全局日志打印级别
    log-level: info
    # 全局日志打印策略
    log-strategy: body
  • 修改日志打印策略后,日志信息更全面了;
  • Retrofit支持四种日志打印策略;
    • NONE:不打印日志;
    • BASIC:只打印日志请求记录;
    • HEADERS:打印日志请求记录、请求和响应头信息;
    • BODY:打印日志请求记录、请求和响应头信息、请求和响应体信息。

全局超时时间

有时候我们需要修改一下Retrofit的请求超时时间,可以通过如下配置实现。

yaml 复制代码
retrofit:
  # 全局超时配置
  global-timeout:
    # 全局连接超时时间
    connect-timeout-ms: 3000
    # 全局读取超时时间
    read-timeout-ms: 3000
    # 全局写入超时时间
    write-timeout-ms: 35000
    # 全局完整调用超时时间
    call-timeout-ms: 0

全局请求重试

  • retrofit-spring-boot-starter支持请求重试,可以通过如下配置实现。
yaml 复制代码
retrofit:
  # 全局重试配置
  global-retry:
    # 是否启用全局重试
    enable: true
    # 全局重试间隔时间
    interval-ms: 100
    # 全局最大重试次数
    max-retries: 2
    # 全局重试规则
    retry-rules:
      - response_status_not_2xx
      - occur_exception
  • 重试规则retry-rules支持如下三种配置。
    • RESPONSE_STATUS_NOT_2XX:响应状态码不是2xx时执行重试;
    • OCCUR_IO_EXCEPTION:发生IO异常时执行重试;
    • OCCUR_EXCEPTION:发生任意异常时执行重试。

总结

今天体验了一把Retrofit,对比使用HttpUtil,确实优雅不少!通过接口发起HTTP请求已不再是Feign的专属,通过Retrofit我们在单体应用中照样可以使用这种方式。当然retrofit-spring-boot-starter提供的功能远不止于此,它还能支持微服务间的调用和熔断降级,感兴趣的朋友可以研究下!

参考资料

官方文档:github.com/LianjiaTech...

项目源码地址

github.com/macrozheng/...

相关推荐
武子康几秒前
大数据-57 Kafka 高级特性 Producer 消息发送流程与核心配置详解
大数据·后端·kafka
知其然亦知其所以然2 分钟前
MySQL社招面试题:索引有哪几种类型?我讲给你听的不只是答案!
后端·mysql·面试
天天摸鱼的java工程师5 分钟前
掘金图片上传被拒:一次由CheckAuthenticationError引发的密钥‘失踪’迷案
java·后端
福大大架构师每日一题6 分钟前
2025-08-01:粉刷房子Ⅳ。用go语言,给定一个偶数个房屋排列在一条直线上,和一个大小为 n x 3 的二维数组 cost,其中 cost[i][j] 表
后端
error_cn6 分钟前
网络i_o对cpu负载分析
后端
用户053140608817 分钟前
Spring Boot AOP 切点表达式深度解析
spring boot
bug菌7 分钟前
学生信息管理系统,真的是码农的必修课吗?
java·后端·java ee
就是帅我不改9 分钟前
深入实战建造者模式:在订单系统中的应用
后端·架构
李长渊哦10 分钟前
SpringBoot中ResponseEntity的使用详解
java·spring boot·后端
王中阳Go10 分钟前
灵活分库分表,面试的时候这么说,加分!
后端·面试