1.問題梳理:
異常:org.springframework.web.HttpRequestMethodNotSupportedException: Request method 'POST' not supported
很明顯是最終feign執行http請求時把這個方法認定為POST,但feign client中又定義了RequestMethod.GET 或 @GetMapping,沖突導致報錯
那么為什么feign會認為這個方法是post呢?
源碼追蹤:
1.我們從feignClient注解作為入口來看:
2.按照spring cloud一貫風格,我們打開FeignAutoConfiguration這個類看配置邏輯:
看上圖紅框內的標注:這個類在不存在ILoadBalancer時才觸發,我們項目開啟了Ribbon,所以肯定存在,再看注釋:載入負載均衡ribbon clients需要走FeignRibbonClientAutoConfiguration這個類配置。go~
如上圖,看紅框注釋:按照導入從上往下的順序:HttpClientFeignLoadBalancedConfiguration>OkHttpFeignLoadBalancedConfiguration>DefaultFeignLoadBalancedConfiguration,對應的底層http工具:httpclient>okhttp>HttpURLConnection
根據http協議定義是支持@RequestBody+ RequestMethod.GET的,那么具體就得看工具包實現的不同的,查看源碼發現okhttp和HttpURLConnection都不支持(報錯),只有httpclient支持。(默認走HttpURLConnection會報錯)
我們知道只有httpclient支持@RequestBody+ RequestMethod.GET,所以我們必須滿足條件走HttpClientFeignLoadBalancedConfiguration才行,看下源碼:
可見,滿足類路徑下存在ApacheHttpClient類即可。我們再pom中添加:
<dependency> <groupId>io.github.openfeign</groupId> <artifactId>feign-httpclient</artifactId> </dependency>
最終載入了feign-httpclient-9.5.0.jar包,打開看里面就一個ApacheHttpClient.class,點進去看其實就是一個httpclient。
所以pom中引入feign-httpclient--》類路徑下存在ApacheHttpClient.class--》走HttpClientFeignLoadBalancedConfiguration--》請求時走HttpClient--》支持@RequestBody+ RequestMethod.GET
2.解決方式:
pom中引入
<dependency> <groupId>io.github.openfeign</groupId> <artifactId>feign-httpclient</artifactId> </dependency>
maven更新后查看項目中是否存在了feign-httpclient-9.5.0.jar包。