Spring cloud踩坑記錄之使用feignclient遠(yuǎn)程調(diào)用服務(wù)404的方法
前言
公司項目進(jìn)行微服務(wù)改造,由之前的dubbo改用SpringCloud,微服務(wù)之間通過FeignClient進(jìn)行調(diào)用,今天在測試的時候,eureka注冊中心有相應(yīng)的服務(wù),但feignclient就是無法調(diào)通,一直報404錯誤,排查過程如下:
一、問題:
服務(wù)提供方定義的接口如下:
/** * 黑白名單查詢接口 * * @author LiJunJun * @since 2018/10/18 */ @Component(value = "blackAndWhiteListFeignClient") @FeignClient(value = "pear-cache-service", path = "v1/cache/limitlist") public interface IBlackAndWhiteListFeignClient { /** * 獲取黑白名單手機號分組編號 * * @param trace 請求流水 * @param phoneNum 電話號碼 * @return 電話號碼所在分組 */ @RequestMapping(value = "/blackAndWhiteList", method = RequestMethod.POST, produces = MediaType.APPLICATION_JSON_UTF8_VALUE) ResultData<String> blackAndWhiteList(@RequestHeader(name = "Trace") String trace, @RequestParam("phoneNum") String phoneNum); }
接口實現(xiàn)類如下:
/** * 黑白名單controller * * @author LiJunJun * @since 2018/10/18 */ @ProtectedLdApi @RestController @RequestMapping(value = "v1/cache/limitlist") @Api(value = "黑白名單緩存", description = "黑白名單緩存相關(guān)接口") public class BlacklAndWhiteListController extends AbstractController implements IBlackAndWhiteListFeignClient { /** * 日志記錄器 */ private final static Log LOGGER = new Log(BlacklAndWhiteListController.class); /** * 注入tedis */ @Autowired private JedisSentinelPoolExt jedisSentinelPool; /** * 獲取黑白名單手機號分組編號 * * @param trace 請求流水 * @param phoneNum 電話號碼 * @return 電話號碼所在分組 */ @Override @ApiOperation(value = "獲取黑白名單手機號分組編號", notes = "根據(jù)電話號碼從緩存中獲取黑白名單分組") @RequestMapping(value = "/blackAndWhiteList", method = RequestMethod.POST, consumes = MediaType.APPLICATION_JSON_UTF8_VALUE, produces = MediaType.APPLICATION_JSON_UTF8_VALUE) public ResultData<String> blackAndWhiteList(@RequestHeader(name = "Trace") String trace, @RequestParam("phoneNum") String phoneNum) { do something... } }
調(diào)用方如下:
public class MessageListController { private static final Log LOGGER = new Log(MessageListController.class); @Autowired private IBlackAndWhiteListFeignClient blackAndWhiteListFeignClient; @RequestMapping(value = "/testBlackAndWhiteList", method = RequestMethod.POST, produces = MediaType.APPLICATION_JSON_UTF8_VALUE) public ResultData<String> testBlackAndWhiteList() { LOGGER.info("開始調(diào)用緩存接口"); ResultData<String> res = blackAndWhiteListFeignClient.blackAndWhiteList("asdqwezxxc", "B18037903086"); LOGGER.info("調(diào)用結(jié)果:" + res.getResultData()); return res; }
調(diào)用結(jié)果:
華麗麗的404了,很頭疼,經(jīng)過各種度娘,發(fā)現(xiàn)導(dǎo)致這個問題有兩個原因,以下是解決方法:
二、問題分析
經(jīng)過百度,說將SpringBoot配置文件里面 server.servlet.context-path
注釋掉即可,抱著試一哈的態(tài)度,注釋了,重啟,調(diào)用,結(jié)果驚喜的發(fā)現(xiàn),依舊報錯了,但仔細(xì)一看,錯誤代碼已經(jīng)不是404,變成了415,這就相當(dāng)于調(diào)通了,但是,Content-Type的類型不對,于是,返回去看代碼(此時已經(jīng)肯定,今天能把feignclient接口調(diào)通),
仔細(xì)一看發(fā)現(xiàn),接口上定義的@RequestMapping中,只定義了 produces = MediaType.APPLICATION_JSON_UTF8_VALUE
,而實現(xiàn)類中,@RequestMapping定義了consumes、produces均為 "application/json;charset=UTF-8"
我們知道,consumes定義了方法接受的Http的請求類型,produces則定義了Http請求返回的類型;
然后我們說下FeignClient,它的底層實現(xiàn),就是根據(jù)定義的FeignClient接口,來組裝Http請求進(jìn)行遠(yuǎn)程調(diào)用,而Http默認(rèn)的Content-type是x-www-form-urlencoded類型化的,到這兒,問題就呼之欲出了:
再來回顧上面我們定義的接口,并沒有指定請求類型(consumes),那么FeignClient組裝的Http請求的類型就是默認(rèn)的x-www-form-urlencoded類型,但我們的實現(xiàn)類上,卻定義了consumes=MediaType.APPLICATION_JSON_UTF8_VALUE
,也就是說,僅接受json類型的請求,這就是為什么415的原因了;
三、解決方法
知道了問題的原因,解決起來就很簡單了,我們可以在FeignClient的接口定義上,指定consumes,這樣,F(xiàn)eignClient在組裝Http請求的時候,就會在header里面設(shè)置請求類型為application/json,這樣,問題就完美解決;
再來看調(diào)用結(jié)果:
完美返回?。?!
四、總結(jié)
feignclient接口定義是一個模板化的,其組裝的Http請求完全按照你定義的接口去組裝,如你在參數(shù)中,用@RequestHeader去接收一個參數(shù),其組裝請求時,就會將你傳入的參數(shù)放至Header中,你指定的consumes為json,其組裝的請求Content-Type就是 application/json類型的,完全不需要調(diào)用方感知,就像調(diào)用普通方法一樣,不得不說,很強大,只要生成的Http請求正確,服務(wù)提供方提供的Rest接口能和FeignClient組裝的Http請求,就能夠完成遠(yuǎn)程調(diào)用。
五、遺留問題
為什么需要將服務(wù)提供方的server.servlet.context-path
去掉才能實現(xiàn)調(diào)用,今天暫時沒有研究,但一定有解決方案,SpringCloud不會這么low的,解決方案研究出來會補上。
好了,以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,如果有疑問大家可以留言交流,謝謝大家對腳本之家的支持。
相關(guān)文章
SpringData JPA Mongodb查詢部分字段問題
這篇文章主要介紹了SpringData JPA Mongodb查詢部分字段問題,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2024-08-08Java面試??贾瓹oncurrentHashMap多線程擴容機制詳解
幾乎所有的后端技術(shù)面試官都要在?ConcurrentHashMap?技術(shù)的使用和原理方面對小伙伴們進(jìn)行刁難,本文主要來和大家聊聊ConcurrentHashMap多線程的擴容機制,希望對大家有所幫助2023-05-05SpringBoot超詳細(xì)講解Thymeleaf模板引擎
這篇文章主要分享了Spring Boot整合使用Thymeleaf,Thymeleaf是新一代的Java模板引擎,類似于Velocity、FreeMarker等傳統(tǒng)引擎,關(guān)于其更多相關(guān)內(nèi)容,需要的小伙伴可以參考一下2022-07-07SpringDataJpa:JpaRepository增刪改查操作
這篇文章主要介紹了SpringDataJpa:JpaRepository增刪改查操作,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-08-08redis之基于SpringBoot實現(xiàn)Redis stream實時流事件處理方式
這篇文章主要介紹了redis之基于SpringBoot實現(xiàn)Redis stream實時流事件處理方式,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2023-06-06Java線程安全解決方案(synchronized,ReentrantLock,Atomic)
這篇文章主要介紹了Java線程安全解決方案(synchronized,ReentrantLock,Atomic),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-09-09利用java監(jiān)聽器實現(xiàn)在線人數(shù)統(tǒng)計
過去使用ASP和ASP.NET兩種編程的時候,都寫過在線人數(shù)統(tǒng)計能,實現(xiàn)功能挺簡單的!今天使用java來實現(xiàn)在線人數(shù)統(tǒng)計有點另類,是通過Java監(jiān)聽器實現(xiàn)的,需要的朋友可以參考下2015-09-09Spring?Boot使用MyBatis進(jìn)行兩個表的關(guān)聯(lián)
本文主要介紹了Spring?Boot使用MyBatis進(jìn)行兩個表的關(guān)聯(lián),通過實例演示了如何使用MyBatis的XML映射文件和注解實現(xiàn)關(guān)聯(lián)操作,具有一定的參考價值,感興趣的可以了解一下2023-09-09