SpringBoot動態(tài)Feign服務(wù)調(diào)用詳解
1.Feign傳統(tǒng)方式的不足
①.在微服務(wù)架構(gòu)中,當(dāng)我們使用Feign傳統(tǒng)方式進行服務(wù)調(diào)用的時候,需要在每個服務(wù)消費者中添加FeignClient接口,編寫對應(yīng)的方法,而且當(dāng)服務(wù)生產(chǎn)者Handler新增方法之后,服務(wù)消費者也要在FeignClient接口中添加方法,這樣的話,會有些累贅.
那么能不能在調(diào)用服務(wù)提供者方法的時候,傳入生產(chǎn)者服務(wù)名稱的動態(tài)生成FeignClient對象,然后通過一個通用的方法,指定Handler的URL以及參數(shù)就可以調(diào)用到指定的方法呢?
當(dāng)然可以!
2.動態(tài)Feign
2.1.服務(wù)生產(chǎn)者
由于微服務(wù)項目創(chuàng)建過程并不是重點,所以后續(xù)只展示關(guān)鍵的代碼.
Handler方法
@RestController
public class OrderController {
// 模擬數(shù)據(jù)庫存儲數(shù)據(jù)
private static Map<String, Object> maps = new HashMap<>(10);
static {
maps.put("1", "筆記本");
}
@GetMapping(value = "/getOrderById/{oId}")
public Object getOrderById(@PathVariable(name = "oId") String oid) {
return maps.get(oid);
}
@PostMapping(value = "/addOrder")
public boolean addOrder(@RequestBody Map<String, String> orderMap) {
try {
maps.putAll(orderMap);
return true;
} catch (Exception e) {
e.printStackTrace();
return false;
}
}
}
2.2.動態(tài)Feign
為了方便在多個服務(wù)消費者模塊中使用動態(tài)Feign的功能,可以將動態(tài)Feign相關(guān)的功能單獨抽取出來作為一個公共的模塊,之后在服務(wù)消費者模塊中直接依賴該公共模塊即可.
①.pom.xml
<dependencies>
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-openfeign</artifactId>
</dependency>
</dependencies>
②.通用方法
/**
* Description: 通用接口,里面定義通用方法
* 注意: 由于服務(wù)生產(chǎn)者所有的接口的返回值都是json格式的字符串,
* 所以這里的通用接口的返回值最好使用String類型!!!
*/
public interface DynamicService {
/**
* post方法對應(yīng)的方法
* @param url 服務(wù)生產(chǎn)者Handler方法請求映射地址
* @param params 服務(wù)生產(chǎn)者Handler方法參數(shù)
* @return
*/
@PostMapping(value = "{url}")
String executePostRequest(@PathVariable("url") String url, @RequestBody Object params);
@GetMapping(value = "{url}")
String executeGetRequest(@PathVariable("url") String url, @SpringQueryMap Object params);
@PutMapping(value = "{url}")
String executePutRequest(@PathVariable("url") String url, @RequestBody Object params);
@DeleteMapping(value = "{url}")
String executeDeleteRequest(@PathVariable("url") String url, @RequestBody Object params);
}
②.Feign工廠類
/**
* Description: FeignClient工廠類,根據(jù)服務(wù)名稱創(chuàng)建FeignClient對象(代理對象)
*/
@Component
public class DynamicFeignClientFactory <T>{
private FeignClientBuilder feignClientBuilder;
public DynamicFeignClientFactory(ApplicationContext applicationContext){
this.feignClientBuilder = new FeignClientBuilder(applicationContext);
}
//動態(tài)生成feignClient對象(代理對象)
public T getFeignClient(final Class<T> type,String ServiceID){
return this.feignClientBuilder.forType(type,ServiceID).build();
}
}③.動態(tài)FeignClient類
/**
* Description: 通過FeignClient工廠獲取到的FeignClient對象通過指定的請求去調(diào)用生產(chǎn)者方法!
*/
@Component
public class DynamicClient {
@Autowired
private DynamicFeignClientFactory<DynamicService> dynamicDynamicFeignClientFactory;
public Object executePostApi(String feignName,String url,Object params){
DynamicService dynamicService = dynamicDynamicFeignClientFactory.getFeignClient(DynamicService.class,feignName);
return dynamicService.executePostRequest(url,params);
}
public Object executeGetApi(String feignName,String url,Object params){
DynamicService dynamicService = dynamicDynamicFeignClientFactory.getFeignClient(DynamicService.class,feignName);
return dynamicService.executeGetRequest(url, params);
}
//省略了兩個方法...
}動態(tài)Feign一旦定義好之后,不管后續(xù)服務(wù)生產(chǎn)者中添加加多少Handler方法,這里的代碼基本上不需要修改,對于服務(wù)消費者來說也不會產(chǎn)生任何影響(有點類似于設(shè)計模式里面的"開閉原則");
2.3.服務(wù)消費者
①.引入動態(tài)Feign模塊
<dependency>
<groupId>com.xp</groupId>
<artifactId>dynamic-feign</artifactId>
<version>1.0-SNAPSHOT</version>
</dependency>
②.Handler接口
@RestController
public class UserController {
//注入動態(tài)FeignClient對象
@Autowired
private DynamicClient dynamicClient;
@GetMapping(value = "/getorder/{oid}")
public Object getOrderById(@PathVariable(value = "oid") String oid) {
//通過動態(tài)FeignClient,指定服務(wù)名稱,Handler方法URL,參數(shù)等信息即可調(diào)用生產(chǎn)者對應(yīng)的方法
return this.dynamicClient.executeGetApi("cloud-order", "/getOrderById/".concat(oid),new HashMap<>());
}
@PostMapping(value = "/addOrder")
public Object addOrder(@RequestBody Map<String,Object> map){
return this.dynamicClient.executePostApi("cloud-order", "/addOrder", map);
}
}③.啟動類
@SpringBootApplication
@EnableDiscoveryClient //服務(wù)治理
@EnableFeignClients //開啟Feign功能(不寫也可以!)
public class UserApplication {
public static void main(String[] args) {
SpringApplication.run(UserApplication.class,args);
}
}
代碼完成之后,啟動相關(guān)的服務(wù),就可以進行測試

3.總結(jié)
使用動態(tài)Feign方式進行服務(wù)調(diào)用可以讓開發(fā)者少寫很多代碼,使其可以專注于業(yè)務(wù)本身!
到此這篇關(guān)于SpringBoot動態(tài)Feign服務(wù)調(diào)用詳解的文章就介紹到這了,更多相關(guān)SpringBoot Feign內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
IDEA啟動Tomcat時控制臺出現(xiàn)亂碼問題及解決
這篇文章主要介紹了IDEA啟動Tomcat時控制臺出現(xiàn)亂碼問題及解決方案,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2024-02-02
Jboss Marshalling服務(wù)端無法接受消息
這篇文章主要介紹了Jboss Marshalling服務(wù)端無法接受消息,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下2020-03-03

