前言
公司項目進行微服務改造,由之前的dubbo改用springcloud,微服務之間通過feignclient進行調用,今天在測試的時候,eureka注冊中心有相應的服務,但feignclient就是無法調通,一直報404錯誤,排查過程如下:
一、問題:
服務提供方定義的接口如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
|
/** * 黑白名單查詢接口 * * @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); } |
接口實現類如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
|
/** * 黑白名單controller * * @author lijunjun * @since 2018/10/18 */ @protectedldapi @restcontroller @requestmapping (value = "v1/cache/limitlist" ) @api (value = "黑白名單緩存" , description = "黑白名單緩存相關接口" ) 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 = "根據電話號碼從緩存中獲取黑白名單分組" ) @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... } } |
調用方如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
|
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( "開始調用緩存接口" ); resultdata<string> res = blackandwhitelistfeignclient.blackandwhitelist( "asdqwezxxc" , "b18037903086" ); logger.info( "調用結果:" + res.getresultdata()); return res; } |
調用結果:
華麗麗的404了,很頭疼,經過各種度娘,發現導致這個問題有兩個原因,以下是解決方法:
二、問題分析
經過百度,說將springboot配置文件里面 server.servlet.context-path
注釋掉即可,抱著試一哈的態度,注釋了,重啟,調用,結果驚喜的發現,依舊報錯了,但仔細一看,錯誤代碼已經不是404,變成了415,這就相當于調通了,但是,content-type的類型不對,于是,返回去看代碼(此時已經肯定,今天能把feignclient接口調通),
仔細一看發現,接口上定義的@requestmapping中,只定義了 produces = mediatype.application_json_utf8_value
,而實現類中,@requestmapping定義了consumes、produces均為 "application/json;charset=utf-8"
我們知道,consumes定義了方法接受的http的請求類型,produces則定義了http請求返回的類型;
然后我們說下feignclient,它的底層實現,就是根據定義的feignclient接口,來組裝http請求進行遠程調用,而http默認的content-type是x-www-form-urlencoded類型化的,到這兒,問題就呼之欲出了:
再來回顧上面我們定義的接口,并沒有指定請求類型(consumes),那么feignclient組裝的http請求的類型就是默認的x-www-form-urlencoded類型,但我們的實現類上,卻定義了consumes=mediatype.application_json_utf8_value
,也就是說,僅接受json類型的請求,這就是為什么415的原因了;
三、解決方法
知道了問題的原因,解決起來就很簡單了,我們可以在feignclient的接口定義上,指定consumes,這樣,feignclient在組裝http請求的時候,就會在header里面設置請求類型為application/json,這樣,問題就完美解決;
再來看調用結果:
完美返回!!!
四、總結
feignclient接口定義是一個模板化的,其組裝的http請求完全按照你定義的接口去組裝,如你在參數中,用@requestheader去接收一個參數,其組裝請求時,就會將你傳入的參數放至header中,你指定的consumes為json,其組裝的請求content-type就是 application/json類型的,完全不需要調用方感知,就像調用普通方法一樣,不得不說,很強大,只要生成的http請求正確,服務提供方提供的rest接口能和feignclient組裝的http請求,就能夠完成遠程調用。
五、遺留問題
為什么需要將服務提供方的server.servlet.context-path
去掉才能實現調用,今天暫時沒有研究,但一定有解決方案,springcloud不會這么low的,解決方案研究出來會補上。
好了,以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作具有一定的參考學習價值,如果有疑問大家可以留言交流,謝謝大家對服務器之家的支持。
原文鏈接:https://www.cnblogs.com/ft535535/p/9898147.html