Spring cloud踩坑記錄之使用feignclient遠(yuǎn)程調(diào)用服務(wù)404的方法
前言
公司項(xiàng)目進(jìn)行微服務(wù)改造,由之前的dubbo改用SpringCloud,微服務(wù)之間通過FeignClient進(jìn)行調(diào)用,今天在測(cè)試的時(shí)候,eureka注冊(cè)中心有相應(yīng)的服務(wù),但feignclient就是無法調(diào)通,一直報(bào)404錯(cuò)誤,排查過程如下:
一、問題:
服務(wù)提供方定義的接口如下:
/**
* 黑白名單查詢接口
*
* @author LiJunJun
* @since 2018/10/18
*/
@Component(value = "blackAndWhiteListFeignClient")
@FeignClient(value = "pear-cache-service", path = "v1/cache/limitlist")
public interface IBlackAndWhiteListFeignClient {
/**
* 獲取黑白名單手機(jī)號(hào)分組編號(hào)
*
* @param trace 請(qǐng)求流水
* @param phoneNum 電話號(hào)碼
* @return 電話號(hào)碼所在分組
*/
@RequestMapping(value = "/blackAndWhiteList", method = RequestMethod.POST, produces = MediaType.APPLICATION_JSON_UTF8_VALUE)
ResultData<String> blackAndWhiteList(@RequestHeader(name = "Trace") String trace, @RequestParam("phoneNum") String phoneNum);
}
接口實(shí)現(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;
/**
* 獲取黑白名單手機(jī)號(hào)分組編號(hào)
*
* @param trace 請(qǐng)求流水
* @param phoneNum 電話號(hào)碼
* @return 電話號(hào)碼所在分組
*/
@Override
@ApiOperation(value = "獲取黑白名單手機(jī)號(hào)分組編號(hào)", notes = "根據(jù)電話號(hào)碼從緩存中獲取黑白名單分組")
@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)致這個(gè)問題有兩個(gè)原因,以下是解決方法:
二、問題分析
經(jīng)過百度,說將SpringBoot配置文件里面 server.servlet.context-path 注釋掉即可,抱著試一哈的態(tài)度,注釋了,重啟,調(diào)用,結(jié)果驚喜的發(fā)現(xiàn),依舊報(bào)錯(cuò)了,但仔細(xì)一看,錯(cuò)誤代碼已經(jīng)不是404,變成了415,這就相當(dāng)于調(diào)通了,但是,Content-Type的類型不對(duì),于是,返回去看代碼(此時(shí)已經(jīng)肯定,今天能把feignclient接口調(diào)通),
仔細(xì)一看發(fā)現(xiàn),接口上定義的@RequestMapping中,只定義了 produces = MediaType.APPLICATION_JSON_UTF8_VALUE,而實(shí)現(xiàn)類中,@RequestMapping定義了consumes、produces均為 "application/json;charset=UTF-8"
我們知道,consumes定義了方法接受的Http的請(qǐng)求類型,produces則定義了Http請(qǐng)求返回的類型;
然后我們說下FeignClient,它的底層實(shí)現(xiàn),就是根據(jù)定義的FeignClient接口,來組裝Http請(qǐng)求進(jìn)行遠(yuǎn)程調(diào)用,而Http默認(rèn)的Content-type是x-www-form-urlencoded類型化的,到這兒,問題就呼之欲出了:
再來回顧上面我們定義的接口,并沒有指定請(qǐng)求類型(consumes),那么FeignClient組裝的Http請(qǐng)求的類型就是默認(rèn)的x-www-form-urlencoded類型,但我們的實(shí)現(xiàn)類上,卻定義了consumes=MediaType.APPLICATION_JSON_UTF8_VALUE,也就是說,僅接受json類型的請(qǐng)求,這就是為什么415的原因了;
三、解決方法
知道了問題的原因,解決起來就很簡單了,我們可以在FeignClient的接口定義上,指定consumes,這樣,F(xiàn)eignClient在組裝Http請(qǐng)求的時(shí)候,就會(huì)在header里面設(shè)置請(qǐng)求類型為application/json,這樣,問題就完美解決;

再來看調(diào)用結(jié)果:

完美返回?。?!
四、總結(jié)
feignclient接口定義是一個(gè)模板化的,其組裝的Http請(qǐng)求完全按照你定義的接口去組裝,如你在參數(shù)中,用@RequestHeader去接收一個(gè)參數(shù),其組裝請(qǐng)求時(shí),就會(huì)將你傳入的參數(shù)放至Header中,你指定的consumes為json,其組裝的請(qǐng)求Content-Type就是 application/json類型的,完全不需要調(diào)用方感知,就像調(diào)用普通方法一樣,不得不說,很強(qiáng)大,只要生成的Http請(qǐng)求正確,服務(wù)提供方提供的Rest接口能和FeignClient組裝的Http請(qǐng)求,就能夠完成遠(yuǎn)程調(diào)用。
五、遺留問題
為什么需要將服務(wù)提供方的server.servlet.context-path去掉才能實(shí)現(xiàn)調(diào)用,今天暫時(shí)沒有研究,但一定有解決方案,SpringCloud不會(huì)這么low的,解決方案研究出來會(huì)補(bǔ)上。
好了,以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,如果有疑問大家可以留言交流,謝謝大家對(duì)腳本之家的支持。
相關(guān)文章
Java數(shù)據(jù)結(jié)構(gòu)之LinkedList的用法詳解
鏈表(Linked?list)是一種常見的基礎(chǔ)數(shù)據(jù)結(jié)構(gòu),是一種線性表。Java的LinkedList(鏈表)?類似于?ArrayList,是一種常用的數(shù)據(jù)容器,本文就來簡單講講它的使用吧2023-05-05
Eclipse maven項(xiàng)目lombok安裝配置圖解
這篇文章主要介紹了Eclipse maven項(xiàng)目lombok安裝配置圖解,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-05-05
Java學(xué)習(xí)常用包(類)之java.util包詳解
這篇文章主要介紹了Java學(xué)習(xí)常用包(類)之java.util包的相關(guān)資料,Java.util包是Java標(biāo)準(zhǔn)類庫的重要組成部分,包含集合框架、日期時(shí)間類、事件模型、隨機(jī)數(shù)生成器等實(shí)用工具類,集合框架提供了多種數(shù)據(jù)結(jié)構(gòu)和算法,需要的朋友可以參考下2024-10-10
關(guān)于request.getRequestDispatcher().forward()的妙用及DispatcherType
這篇文章主要介紹了關(guān)于request.getRequestDispatcher().forward()的妙用及DispatcherType對(duì)Filter配置的影響,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-01-01
使用CORS實(shí)現(xiàn)JavaWeb跨域請(qǐng)求問題的方法
這篇文章主要介紹了使用Cors實(shí)現(xiàn)JavaWeb跨域請(qǐng)求問題的方法,非常不錯(cuò),具有參考借鑒價(jià)值,需要的朋友可以參考下2016-09-09

