您好,登錄后才能下訂單哦!
這篇文章主要介紹了spring如何根據controller中接收請求參數不同service,具有一定借鑒價值,感興趣的朋友可以參考下,希望大家閱讀完這篇文章之后大有收獲,下面讓小編帶著大家一起了解一下。
附controller的分類:
正文
我的解決辦法是,使用“生產線工人工作能力自己掂量機制”來解決,這名字我自己起的,實際上就是想要實現按參數選擇走哪個 service 實現,可以一次性把所有 service 實現全都注入進來,然后依次請求,同時在每個 service 實現中寫一套規則判別方法,判斷當前請求自己是不是能夠處理,如果能夠處理,則進入處理方法,若自己沒有處理能力,則退出,讓請求走到其他 service 做同樣的判斷。形象點,可以想象一下,在一條生產線的傳送帶上傳送著不同品類的待加工的元部件,有若干工人排列在傳送帶旁邊,每個工人只會加工某一種元件,那么,當傳送帶上的元件傳送到自己面前時,需要判斷一下,自己有沒有處理這個元件的能力(掂量一下自己的能力),若有,取過來處理,若沒有,放過去讓別人走流程。
理解了其中邏輯,我們就來看代碼吧(片段):
public interface ServiceProvider { // 掂量一下自己有沒有能力加工當前的元件(也就是Request),能就返回 true 不能返回 false boolean support(Request request); // 具體加工元件的邏輯 Response execute(Request request); }
@Service public class ServiceImpl implements Service { // 注入一系列 service 數量不定 https://stackoverflow.com/questions/2153298/how-to-autowire-factorybean @Resource(name = "serviceProviders") private List<ServiceProvider> serviceProviders; @Override public List<Response> execute(Request request) { return serviceProviders // 循環每個 service TODO 現在時間復雜度為 O(n) 可以嘗試優化為 O(logn) .stream() .filter(serviceProvider -> serviceProvider.support(request)) // 按加工能力過濾 .map(serviceProvider -> serviceProvider.execute(request)) // 執行 service 得 execute 方法 .collect(Collectors.toList()); } }
這里有一點需要解釋一下,在上面第二段代碼中,有這么一段:
@Resource(name = "serviceProviders") private List<ServiceProvider> serviceProviders;
這里是使用 spring 中的 FactoryBean 機制實現的,可以簡單的這樣理解 FactoryBean :FactoryBean 是生成普通 Bean 的 Bean,當注入 FactoryBean 時,默認注入的是其生產出來的所有普通 Bean,而不是它自己。
在上邊代碼中,注入的名為 serviceProviders 的這個 Bean,實際上是這樣定義出來的:
@Component("serviceProviders") // 注意這個 Bean 的名字,當其他 Bean 中注入這個 Bean 時,會注入 createInstance() 返回類型的 Bean,而不是其自身的類型 ServiceProviderFactoryBean public class ServiceProviderFactoryBean extends AbstractFactoryBean<List<ServiceProvider>> implements ApplicationContextAware { private ApplicationContext applicationContext; @Override public Class<?> getObjectType() { return List.class; } @Override protected List<ServiceProvider> createInstance() { // 掃描所有 provider 并從 Bean 容器取出放入 list Reflections reflections = new Reflections(ServiceProvider.class.getPackage().getName()); return reflections .getSubTypesOf(ServiceProvider.class) .stream() .map((Function<Class<? extends ServiceProvider>, ServiceProvider>) serviceProviderClass -> applicationContext.getBean(serviceProviderClass)) .collect(Collectors.toList()); } @Override public void setApplicationContext(ApplicationContext applicationContext) throws BeansException { this.applicationContext = applicationContext; } }
通過這樣的設計,就完成了我們的需求,實際上我們等于把思路反轉了一下,從想盡辦法控制注入到不做控制,一股腦全部注入進去,然后按規則過濾。有時候,其實遇到一條思路走不通的時候,可以反過來想想,也許就會走通。
感謝你能夠認真閱讀完這篇文章,希望小編分享的“spring如何根據controller中接收請求參數不同service”這篇文章對大家有幫助,同時也希望大家多多支持億速云,關注億速云行業資訊頻道,更多相關知識等著你來學習!
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。