亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

SpringBoot中使用HTTP客戶端工具Retrofit

 更新時間:2022年06月14日 14:30:23   作者:macrozheng  
這篇文章主要為大家介紹了SpringBoot中使用HTTP客戶端工具Retrofit方法詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪

前言

我們平時開發(fā)項目時,就算是單體應(yīng)用,也免不了要調(diào)用一下其他服務(wù)提供的接口。此時就會用到HTTP客戶端工具,之前一直使用的是Hutool中的HttpUtil,雖然容易上手,但用起來頗為麻煩!最近發(fā)現(xiàn)一款更好用的HTTP客戶端工具Retrofit,你只需聲明接口就可發(fā)起HTTP請求,無需進(jìn)行連接、結(jié)果解析之類的重復(fù)操作,用起來夠優(yōu)雅,推薦給大家!

SpringBoot實戰(zhàn)電商項目mall(50k+star)地址:https://github.com/macrozheng/mall

簡介

Retrofit是適用于AndroidJava且類型安全的HTTP客戶端工具,在Github上已經(jīng)有39k+Star。其最大的特性的是支持通過接口的方式發(fā)起HTTP請求,類似于我們用Feign調(diào)用微服務(wù)接口的那種方式。

SpringBoot是使用最廣泛的Java開發(fā)框架,但是Retrofit官方并沒有提供專門的Starter。于是有位老哥就開發(fā)了retrofit-spring-boot-starter,它實現(xiàn)了Retrofit與SpringBoot框架的快速整合,并且支持了諸多功能增強(qiáng),極大簡化開發(fā)。今天我們將使用這個第三方Starter來操作Retrofit。

使用

在SpringBoot中使用Retrofit是非常簡單的,下面我們就來體驗下。

依賴集成

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

<!--Retrofit依賴-->
<dependency>
    <groupId>com.github.lianjiatech</groupId>
    <artifactId>retrofit-spring-boot-starter</artifactId>
    <version>2.2.18</version>
</dependency>

基本使用

下面以調(diào)用mall-tiny-swagger中的接口為例,我們來體驗下Retrofit的基本使用。

首先我們準(zhǔn)備一個服務(wù)來方便遠(yuǎn)程調(diào)用,使用的是之前的mall-tiny-swagger這個Demo,打開Swagger看下,里面有一個登錄接口和需要登錄認(rèn)證的商品品牌CRUD接口,

項目地址:

https://github.com/macrozheng/mall-learning/tree/master/mall-tiny-swagger

我們先來調(diào)用下登錄接口試試,在application.yml中配置好mall-tiny-swagger的服務(wù)地址;

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

再通過@RetrofitClient聲明一個Retrofit客戶端,由于登錄接口是通過POST表單形式調(diào)用的,這里使用到了@POST和@FormUrlEncoded注解;

/**
 * 定義Http接口,用于調(diào)用遠(yuǎn)程的UmsAdmin服務(wù)
 * Created by macro on 2022/1/19.
 */
@RetrofitClient(baseUrl = "${remote.baseUrl}")
public interface UmsAdminApi {
    @FormUrlEncoded
    @POST("admin/login")
    CommonResult<LoginInfo> login(@Field("username") String username, @Field("password") String password);
}

如果你不太明白這些注解是干嘛的,看下下面的表基本就懂了,更具體的話可以參考Retrofit官方文檔;

接下來在Controller中注入UmsAdminApi,然后進(jìn)行調(diào)用即可;

/**
 * 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 = "調(diào)用遠(yuǎn)程登錄接口獲取token")
    @PostMapping(value = "/admin/login")
    public CommonResult<LoginInfo> login(@RequestParam String username, @RequestParam String password) {
        CommonResult<LoginInfo> result = umsAdminApi.login(username, password);
        LoginInfo loginInfo = result.getData();
        if (result.getData() != null) {
            tokenHolder.putToken(loginInfo.getTokenHead() + " " + loginInfo.getToken());
        }
        return result;
    }
}

為方便后續(xù)調(diào)用需要登錄認(rèn)證的接口,我創(chuàng)建了TokenHolder這個類,把token存儲到了Session中;

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

接下來通過Swagger進(jìn)行測試,調(diào)用接口就可以獲取到遠(yuǎn)程服務(wù)返回的token了,訪問地址:http://localhost:8086/swagger-ui/

注解式攔截器

商品品牌管理接口,需要添加登錄認(rèn)證頭才可以正常訪問,我們可以使用Retrofit中的注解式攔截器來實現(xiàn)。

首先創(chuàng)建一個注解式攔截器TokenInterceptor繼承BasePathMatchInterceptor,然后在doIntercept方法中給請求添加Authorization頭;

/**
 * 給請求添加登錄Token頭的攔截器
 * Created by macro on 2022/1/19.
 */
@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);
    }
}

創(chuàng)建調(diào)用品牌管理接口的客戶端PmsBrandApi,使用@Intercept注解配置攔截器和攔截路徑;

/**
 * 定義Http接口,用于調(diào)用遠(yuǎn)程的PmsBrand服務(wù)
 * Created by macro on 2022/1/19.
 */
@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實例,并添加方法調(diào)用遠(yuǎn)程服務(wù)即可;

/**
 * Retrofit測試接口
 * Created by macro on 2022/1/19.
 */
@Api(tags = "RetrofitController", description = "Retrofit測試接口")
@RestController
@RequestMapping("/retrofit")
public class RetrofitController {
    @Autowired
    private PmsBrandApi pmsBrandApi;
    @ApiOperation("調(diào)用遠(yuǎn)程接口分頁查詢品牌列表")
    @GetMapping(value = "/brand/list")
    public CommonResult<CommonPage<PmsBrand>> listBrand(@RequestParam(value = "pageNum", defaultValue = "1")
                                                        @ApiParam("頁碼") Integer pageNum,
                                                        @RequestParam(value = "pageSize", defaultValue = "3")
                                                        @ApiParam("每頁數(shù)量") Integer pageSize) {
        return pmsBrandApi.list(pageNum, pageSize);
    }
    @ApiOperation("調(diào)用遠(yuǎn)程接口獲取指定id的品牌詳情")
    @GetMapping(value = "/brand/{id}")
    public CommonResult<PmsBrand> brand(@PathVariable("id") Long id) {
        return pmsBrandApi.detail(id);
    }
    @ApiOperation("調(diào)用遠(yuǎn)程接口添加品牌")
    @PostMapping(value = "/brand/create")
    public CommonResult createBrand(@RequestBody PmsBrand pmsBrand) {
        return pmsBrandApi.create(pmsBrand);
    }
    @ApiOperation("調(diào)用遠(yuǎn)程接口更新指定id品牌信息")
    @PostMapping(value = "/brand/update/{id}")
    public CommonResult updateBrand(@PathVariable("id") Long id, @RequestBody PmsBrand pmsBrand) {
        return pmsBrandApi.update(id,pmsBrand);
    }
    @ApiOperation("調(diào)用遠(yuǎn)程接口刪除指定id的品牌")
    @GetMapping(value = "/delete/{id}")
    public CommonResult deleteBrand(@PathVariable("id") Long id) {
        return  pmsBrandApi.delete(id);
    }
}

在Swagger中調(diào)用接口進(jìn)行測試,發(fā)現(xiàn)已經(jīng)可以成功調(diào)用。

全局?jǐn)r截器

如果你想給所有請求都加個請求頭的話,可以使用全局?jǐn)r截器。

創(chuàng)建SourceInterceptor類繼承BaseGlobalInterceptor接口,然后在Header中添加source請求頭。

/**
 * 全局?jǐn)r截器,給請求添加source頭
 * Created by macro on 2022/1/19.
 */
@Component
public 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的配置很多,下面我們講講日志打印、全局超時時間和全局請求重試這三種最常用的配置。

日志打印 默認(rèn)配置下Retrofit使用basic日志策略,打印的日志非常簡單;

我們可以將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

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

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

  • NONE:不打印日志;
  • BASIC:只打印日志請求記錄;
  • HEADERS:打印日志請求記錄、請求和響應(yīng)頭信息;
  • BODY:打印日志請求記錄、請求和響應(yīng)頭信息、請求和響應(yīng)體信息。

全局超時時間

有時候我們需要修改一下Retrofit的請求超時時間,可以通過如下配置實現(xiàn)。

retrofit:
  # 全局連接超時時間
  global-connect-timeout-ms: 3000
  # 全局讀取超時時間
  global-read-timeout-ms: 3000
  # 全局寫入超時時間
  global-write-timeout-ms: 35000
  # 全局完整調(diào)用超時時間
  global-call-timeout-ms: 0

全局請求重試

retrofit-spring-boot-starter支持請求重試,可以通過如下配置實現(xiàn)。

retrofit:
  # 重試配置
  retry:
    # 是否啟用全局重試
    enable-global-retry: true
    # 全局重試間隔時間
    global-interval-ms: 100
    # 全局最大重試次數(shù)
    global-max-retries: 2
    # 全局重試規(guī)則
    global-retry-rules:
      - response_status_not_2xx
      - occur_exception
    # 重試攔截器
    retry-interceptor: com.github.lianjiatech.retrofit.spring.boot.retry.DefaultRetryInterceptor

重試規(guī)則global-retry-rules支持如下三種配置。

  • RESPONSE_STATUS_NOT_2XX:響應(yīng)狀態(tài)碼不是2xx時執(zhí)行重試;
  • OCCUR_IO_EXCEPTION:發(fā)生IO異常時執(zhí)行重試;
  • OCCUR_EXCEPTION:發(fā)生任意異常時執(zhí)行重試。

總結(jié)

今天體驗了一把Retrofit,對比使用HttpUtil,確實優(yōu)雅不少!通過接口發(fā)起HTTP請求已不再是Feign的專屬,通過Retrofit我們在單體應(yīng)用中照樣可以使用這種方式。當(dāng)然retrofit-spring-boot-starter提供的功能遠(yuǎn)不止于此,它還能支持微服務(wù)間的調(diào)用和熔斷降級,感興趣的朋友可以研究下!

參考資料

官方文檔:https://github.com/LianjiaTech/retrofit-spring-boot-starter

項目源碼地址

https://github.com/macrozheng/mall-learning/tree/master/mall-tiny-retrofit

本文 GitHub https://github.com/macrozheng/mall-learning 已經(jīng)收錄,歡迎大家Star!

更多關(guān)于SpringBoot HTTP客戶端Retrofit的資料請關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • Java_Spring之Spring5?的新特性

    Java_Spring之Spring5?的新特性

    這篇文章主要介紹了Java_Spring中Spring5?的新特性,需要利用jdk8?版本更新的內(nèi)容,依賴庫更新,感興趣的小伙伴可以參考閱讀
    2023-04-04
  • JAVA 枚舉相關(guān)知識匯總

    JAVA 枚舉相關(guān)知識匯總

    這篇文章主要介紹了JAVA 枚舉相關(guān)知識,文中講解的非常詳細(xì),代碼幫助大家更好的參考和學(xué)習(xí),感興趣的朋友可以了解下
    2020-06-06
  • java設(shè)計模式之簡單工廠模式

    java設(shè)計模式之簡單工廠模式

    這篇文章主要為大家詳細(xì)介紹了java設(shè)計模式之簡單工廠模式,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2016-12-12
  • SpringCloud中NacosNamingService的作用詳解

    SpringCloud中NacosNamingService的作用詳解

    這篇文章主要介紹了SpringCloud中NacosNamingService的作用詳解,NacosNamingService類完成服務(wù)實例注冊,撤銷與獲取服務(wù)實例操作,NacosNamingService初始化采用單例模式,使用反射生成,需要的朋友可以參考下
    2023-11-11
  • shiro之INI配置詳解

    shiro之INI配置詳解

    這篇文章主要為大家詳細(xì)介紹了shiro之INI配置的相關(guān)資料,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2017-09-09
  • Java中的FilterOutputStream 簡介_動力節(jié)點Java學(xué)院整理

    Java中的FilterOutputStream 簡介_動力節(jié)點Java學(xué)院整理

    FilterOutputStream 的作用是用來“封裝其它的輸出流,并為它們提供額外的功能”。它主要包括BufferedOutputStream, DataOutputStream和PrintStream。接下來通過本文給大家簡單介紹下FilterOutputStream知識,需要的朋友參考下吧
    2017-05-05
  • Java弱鍵集合WeakHashMap及ConcurrentCache原理詳解

    Java弱鍵集合WeakHashMap及ConcurrentCache原理詳解

    這篇文章主要介紹了Java弱鍵集合WeakHashMap及ConcurrentCache原理詳解,基于哈希表的Map接口實現(xiàn),支持null鍵和值,但是WeakHashMap具有弱鍵,可用來實現(xiàn)緩存存儲,在進(jìn)行GC的時候會自動回收鍵值對,需要的朋友可以參考下
    2023-09-09
  • java學(xué)習(xí)之利用TCP實現(xiàn)的簡單聊天示例代碼

    java學(xué)習(xí)之利用TCP實現(xiàn)的簡單聊天示例代碼

    這篇文章主要給大家介紹了關(guān)于java學(xué)習(xí)筆記之利用TCP實現(xiàn)的簡單聊天的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧。
    2017-12-12
  • 通過代碼理解java泛型

    通過代碼理解java泛型

    本篇文章通過代碼實例讓大家充分的理解java泛型的相關(guān)知識點內(nèi)容,有需要的朋友們可以學(xué)習(xí)下。
    2018-08-08
  • java文件下載設(shè)置中文名稱的實例(response.addHeader)

    java文件下載設(shè)置中文名稱的實例(response.addHeader)

    下面小編就為大家分享一篇java文件下載設(shè)置中文名稱的實例(response.addHeader),具有很好的參考價值,希望對大家有所幫助
    2017-12-12

最新評論