使用spring動態獲取接口的不同實現類
spring動態獲取接口的不同實現類
最近做項目,有個需求是和外部對接,從接口獲取新聞數據,把數據和緩存中的數據對比,多瞭的添加到數據庫,少瞭的刪掉。
接口有兩個,一開始我是在業務類裡寫瞭兩個方法,代碼太長,簡單說就是兩個部分:
public Object saveANews() { //1、獲取A接口新聞列表 //2、和緩存對比,存數據到數據庫 } public Object saveBNews() { //1、獲取B新聞列表 //2、和緩存對比,存數據到數據庫 }
寫完後我發現,因為操作的是數據庫的同一張表,2的部分代碼完全一模一樣,隻有1的部分不同,而1的部分其實就隻有一行代碼。。。
這必須得復用啊,而且是一個業務,也沒必要分別用兩個方法,於是我改成瞭這樣:
//業務接口實現方法 public Object saveNews(NewsUtilService service) { //1、獲取接口新聞列表 List<NewsVO> list = service.queryNews(); //2、和緩存對比,存數據到數據庫 } //定義新聞數據接口 public interface NewsUtilService { List<NewsVO> queryNews(); } //接口的兩個實現類 @Service public class ANewsDataServiceImpl implements NewsUtilService { @Autowired private NewsDataMapper newsDataMapper; @Override public List<NewsVO> queryNews(){ //對接數據 } } @Service public class BNewsDataServiceImpl implements NewsUtilService { @Override public List<NewsVO> queryNews(){ //對接數據 } } //定義工廠類 @Service public class NewsUtilServiceFactory { /** * * Method Name: getNewsUtilService * @param source * @return */ public NewsUtilService getNewsUtilService(String source){ switch(source){ case "a": return new ANewsDataServiceImpl(); case "b": return new BNewsDataServiceImpl(); default: return null; } } } //控制層調用 @RestController public class NewsDataController { @Resource private NewsDataService newsDataService; @Resource private NewsUtilServiceFactory factory; public Object getNewsData(){ String[] sources = {"a","b"}; for (int i = 0; i < sources.length; i++) { NewsUtilService newsUtilService = factory.getNewsUtilService(sources[i]); newsDataService.saveNews(newsUtilService); } } }
本來以為這就大工告成瞭,誰知道運行後控制臺居然報錯瞭:
經過一步步調試,總算發現瞭是什麼問題:
其中一個實現類中註入的Mapper沒有實例化,是null。
一開始我還以為是構造函數調用和註入的順序問題,查瞭半天才明白不是,問題在這裡:
使用new關鍵字實例化的對象不是被spring創建的,不歸spring管,所以A類實現類中Mapper註入的註解根本不生效!
但是因為業務需要,那個mapper又需要用到,怎麼辦呢?
當時想到瞭兩種解決辦法
1、在接口的方法參數裡加入mapper,把mapper作為參數傳進去,但這實在太奇怪瞭,先不說B類實現類根本用不到mapper,而且一個接口定義出來後根本不管它的實現類吧,因為實現類的問題去改接口,,,似乎有點非呀。
於是決定用第二種,修改工廠類變成如下:
//定義工廠類 @Service public class NewsUtilServiceFactory { @Autowired private ANewsDataServiceImpl aNewsDataServiceImpl; @Autowired private BNewsDataServiceImpl bNewsDataServiceImpl; public NewsUtilService getNewsUtilService(String source){ switch(source){ case "a": return aNewsDataServiceImpl; case "b": return bNewsDataServiceImpl; default: return null; } } }
代碼寫出來自己都無語瞭,先把所有的實現類都實例化出來,在根據輸入返回。這不是工廠模式,是商店模式吧。。。但是當時也想不到其他辦法,就先這麼寫瞭,但一直覺得肯定有其他解決方案,直到今天有空,去查瞭一下,才發現自己都多low。。。
其實spring可以動態獲取實現類的~~~
@Service public class NewsUtilServiceFactory { @Autowired private ApplicationContext applicationContext; public NewsUtilService getNewsUtilService(String source){ switch(source){ case "web": return applicationContext.getBean(WebNewsDataServiceImpl.class); case "oa": return applicationContext.getBean(OANewDataServiceImpl.class); default: return null; } } }
這才是正確寫法有木有!
總算弄出來瞭,趕緊記錄下來先~
獲取某接口所有實現類
在springboot項目中,為瞭方便,我們可能需要獲取某一個接口下面的所有實現類,根據名稱進行匹配使用。
正文
1、ServiceLocator.java
package com.yang.config; import com.yang.workOrder.service.IRootService; import org.springframework.beans.BeansException; import org.springframework.context.ApplicationContext; import org.springframework.context.ApplicationContextAware; import org.springframework.stereotype.Component; import java.util.Map; /** * explain:獲取應用上下文並獲取相應的接口實現類 * * @author yang * @date 2021/1/5 */ @Component public class ServiceLocator implements ApplicationContextAware { /** * 用於保存接口實現類名及對應的類 */ private Map<String, IRootService> map; /** * 獲取應用上下文並獲取相應的接口實現類 * @param applicationContext * @throws BeansException */ @Override public void setApplicationContext(ApplicationContext applicationContext) throws BeansException { //根據接口類型返回相應的所有bean map = applicationContext.getBeansOfType(IRootService.class); } /** * 獲取所有實現集合 * @return */ public Map<String, IRootService> getMap() { return map; } /** * 獲取對應服務 * @param key * @return */ public IRootService getService(String key) { return map.get(key); } }
2、IRootService.java
package com.yang.workOrder.service; import com.alibaba.fastjson.JSONObject; import com.yang.workOrder.entity.WorkOrder; /** * explain:基礎流程操作服務接口 * * @author yang * @date 2021/1/5 */ public interface IRootService { /** * 開始流程 * @param workOrder * @return */ boolean startProcess(WorkOrder workOrder); }
3、RootA001ServiceImpl.java
package com.yang.workOrder.service.impl; import com.alibaba.fastjson.JSONObject; import com.yang.workOrder.entity.WorkOrder; import com.yang.workOrder.service.IRootService; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.springframework.stereotype.Service; /** * explain:A_001流程審批實現類 * * @author yang * @date 2021/1/5 */ @Service("A_001") public class RootA001ServiceImpl implements IRootService { private static final Logger LOGGER = LoggerFactory.getLogger(RootA001ServiceImpl.class); @Override public boolean startProcess(WorkOrder workOrder) { return false; } }
4、RootA002ServiceImpl.java
package com.yang.workOrder.service.impl; import com.alibaba.fastjson.JSONObject; import com.yang.workOrder.entity.WorkOrder; import com.yang.workOrder.service.IRootService; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.springframework.stereotype.Service; /** * explain:A_002流程審批實現類 * * @author yang * @date 2021/1/5 */ @Service("A_002") public class RootA002ServiceImpl implements IRootService { private static final Logger LOGGER = LoggerFactory.getLogger(RootA002ServiceImpl.class); @Override public boolean startProcess(WorkOrder workOrder) { return false; } }
結果
以上為個人經驗,希望能給大傢一個參考,也希望大傢多多支持WalkonNet。
推薦閱讀:
- SpringBoot中的main方法註入service
- 解決SpringMvc中普通類註入Service為null的問題
- springboot實現在工具類(util)中調用註入service層方法
- 解決netty中spring對象註入失敗的問題
- Spring運行時手動註入bean的方法實例