SpringBoot中如何使用HTTP客户端工具Retrofit

简介

retrofit是适用于android和java且类型安全的http客户端工具,在github上已经有39k+star。其最大的特性的是支持通过接口的方式发起http请求,类似于我们用feign调用微服务接口的那种方式。

SpringBoot中如何使用HTTP客户端工具Retrofit

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

SpringBoot中如何使用HTTP客户端工具Retrofit

使用

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

依赖集成

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

    com.github.lianjiatech    retrofit-spring-boot-starter    2.2.18

登录后复制

基本使用

下面以调用mall-tiny-swagger中的接口为例,我们来体验下Retrofit的基本使用。

首先我们准备一个服务来方便远程调用,使用的是之前的mall-tiny-swagger这个Demo,打开Swagger看下,里面有一个登录接口和需要登录认证的商品品牌CRUD接口,

SpringBoot中如何使用HTTP客户端工具Retrofit

我们先来调用下登录接口试试,在application.yml中配置好mall-tiny-swagger的服务地址;

remote:  baseUrl: http://localhost:8088/

登录后复制

再通过@RetrofitClient声明一个Retrofit客户端,由于登录接口是通过POST表单形式调用的,这里使用到了@POST和@FormUrlEncoded注解;

/** * 定义Http接口,用于调用远程的UmsAdmin服务 * Created by macro on 2022/1/19. */@RetrofitClient(baseUrl = "${remote.baseUrl}")public interface UmsAdminApi {    @FormUrlEncoded    @POST("admin/login")    CommonResult login(@Field("username") String username, @Field("password") String password);}

登录后复制

如果你不太明白这些注解是干嘛的,看下下面的表基本就懂了,更具体的话可以参考Retrofit官方文档;

SpringBoot中如何使用HTTP客户端工具Retrofit

接下来在Controller中注入UmsAdminApi,然后进行调用即可;

/** * Retrofit测试接口 * Created by macro on 2022/1/19. */@Api(tags = "RetrofitController", description = "Retrofit测试接口")@RestController@RequestMapping("/retrofit")public class RetrofitController {    @Autowired    private UmsAdminApi umsAdminApi;    @Autowired    private TokenHolder tokenHolder;    @ApiOperation(value = "调用远程登录接口获取token")    @PostMapping(value = "/admin/login")    public CommonResult login(@RequestParam String username, @RequestParam String password) {        CommonResult result = umsAdminApi.login(username, password);        LoginInfo loginInfo = result.getData();        if (result.getData() != null) {            tokenHolder.putToken(loginInfo.getTokenHead() + " " + loginInfo.getToken());        }        return result;    }}

登录后复制

为方便后续调用需要登录认证的接口,我创建了TokenHolder这个类,把token存储到了Session中;

/** * 登录token存储(在Session中) * Created by macro on 2022/1/19. */@Componentpublic 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;    }}

登录后复制

接下来通过Swagger进行测试,调用接口就可以获取到远程服务返回的token了,访问地址:http://localhost:8086/swagger-ui/

SpringBoot中如何使用HTTP客户端工具Retrofit

注解式拦截器

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

首先创建一个注解式拦截器TokenInterceptor继承BasePathMatchInterceptor,然后在doIntercept方法中给请求添加Authorization头;

/** * 给请求添加登录Token头的拦截器 * Created by macro on 2022/1/19. */@Componentpublic 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注解配置拦截器和拦截路径;

/** * 定义Http接口,用于调用远程的PmsBrand服务 * Created by macro on 2022/1/19. */@RetrofitClient(baseUrl = "${remote.baseUrl}")@Intercept(handler = TokenInterceptor.class, include = "/brand/**")public interface PmsBrandApi {    @GET("brand/list")    CommonResult> list(@Query("pageNum") Integer pageNum, @Query("pageSize") Integer pageSize);    @GET("brand/{id}")    CommonResult 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实例,并添加方法调用远程服务即可;

/** * Retrofit测试接口 * Created by macro on 2022/1/19. */@Api(tags = "RetrofitController", description = "Retrofit测试接口")@RestController@RequestMapping("/retrofit")public class RetrofitController {    @Autowired    private PmsBrandApi pmsBrandApi;    @ApiOperation("调用远程接口分页查询品牌列表")    @GetMapping(value = "/brand/list")    public CommonResult> listBrand(@RequestParam(value = "pageNum", defaultValue = "1")                                                        @ApiParam("页码") Integer pageNum,                                                        @RequestParam(value = "pageSize", defaultValue = "3")                                                        @ApiParam("每页数量") Integer pageSize) {        return pmsBrandApi.list(pageNum, pageSize);    }    @ApiOperation("调用远程接口获取指定id的品牌详情")    @GetMapping(value = "/brand/{id}")    public CommonResult brand(@PathVariable("id") Long id) {        return pmsBrandApi.detail(id);    }    @ApiOperation("调用远程接口添加品牌")    @PostMapping(value = "/brand/create")    public CommonResult createBrand(@RequestBody PmsBrand pmsBrand) {        return pmsBrandApi.create(pmsBrand);    }    @ApiOperation("调用远程接口更新指定id品牌信息")    @PostMapping(value = "/brand/update/{id}")    public CommonResult updateBrand(@PathVariable("id") Long id, @RequestBody PmsBrand pmsBrand) {        return pmsBrandApi.update(id,pmsBrand);    }    @ApiOperation("调用远程接口删除指定id的品牌")    @GetMapping(value = "/delete/{id}")    public CommonResult deleteBrand(@PathVariable("id") Long id) {        return  pmsBrandApi.delete(id);    }}

登录后复制

在Swagger中调用接口进行测试,发现已经可以成功调用。

SpringBoot中如何使用HTTP客户端工具Retrofit

全局拦截器

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

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

/** * 全局拦截器,给请求添加source头 * Created by macro on 2022/1/19. */@Componentpublic class SourceInterceptor extends BaseGlobalInterceptor {    @Override    protected Response doIntercept(Chain chain) throws IOException {        Request request = chain.request();        Request newReq = request.newBuilder()                .addHeader("source", "retrofit")                .build();        return chain.proceed(newReq);    }}

登录后复制

配置

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

日志打印 默认配置下Retrofit使用basic日志策略,打印的日志非常简单;

SpringBoot中如何使用HTTP客户端工具Retrofit

我们可以将application.yml中的retrofit.global-log-strategy属性修改为body来打印最全日志;

retrofit:  # 日志打印配置  log:    # 启用日志打印    enable: true    # 日志打印拦截器    logging-interceptor: com.github.lianjiatech.retrofit.spring.boot.interceptor.DefaultLoggingInterceptor    # 全局日志打印级别    global-log-level: info    # 全局日志打印策略    global-log-strategy: body

登录后复制

修改日志打印策略后,日志信息更全面了;

SpringBoot中如何使用HTTP客户端工具Retrofit

Retrofit支持四种日志打印策略;

NONE:不打印日志;

BASIC:只打印日志请求记录;

HEADERS:打印日志请求记录、请求和响应头信息;

BODY:打印日志请求记录、请求和响应头信息、请求和响应体信息。

全局超时时间

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

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

登录后复制

全局请求重试

retrofit-spring-boot-starter支持请求重试,可以通过如下配置实现。

retrofit:  # 重试配置  retry:    # 是否启用全局重试    enable-global-retry: true    # 全局重试间隔时间    global-interval-ms: 100    # 全局最大重试次数    global-max-retries: 2    # 全局重试规则    global-retry-rules:      - response_status_not_2xx      - occur_exception    # 重试拦截器    retry-interceptor: com.github.lianjiatech.retrofit.spring.boot.retry.DefaultRetryInterceptor

登录后复制

重试规则global-retry-rules支持如下三种配置。

RESPONSE_STATUS_NOT_2XX:响应状态码不是2xx时执行重试;

OCCUR_IO_EXCEPTION:发生IO异常时执行重试;

OCCUR_EXCEPTION:发生任意异常时执行重试。

以上就是SpringBoot中如何使用HTTP客户端工具Retrofit的详细内容,更多请关注【创想鸟】其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至253000106@qq.com举报,一经查实,本站将立刻删除。

发布者:PHP中文网,转转请注明出处:https://www.chuangxiangniao.com/p/2626988.html

(0)
上一篇 2025年3月7日 00:59:16
下一篇 2025年2月25日 11:37:56

AD推荐 黄金广告位招租... 更多推荐

相关推荐

发表回复

登录后才能评论