來源:JAVA旭陽
不知道大家在項目中有沒有遇到過這樣的場景,根據傳入的類型,調用接口不同的實現類或者說服務,比如根據文件的類型使用 CSV解析器或者JSON解析器,在調用的客戶端一般都是用if else去做判斷,比如類型等于JSON,我就用JSON解析器,那如果新加一個類型的解析器,是不是調用的客戶端還要修改呢?這顯然太耦合了,本文就介紹一種方法,服務定位模式Service Locator Pattern來解決,它幫助我們消除緊耦合實現及其依賴性,并提出將服務與其具體類解耦。
文件解析器的例子
我們通過一個例子來告訴你如何使用Service Locator Pattern。
假設我們有一個從各種來源獲取數據的應用程序,我們必須解析不同類型的文件,比如解析CSV文件和JSON文件。
1、定義一個類型的枚舉
publicenumContentType{ JSON, CSV }
2、定義一個解析的接口
publicinterfaceParser{ Listparse(Readerr); }
3、根據不同的文件類型有不同的實現類
//解析csv @Component publicclassCSVParserimplementsParser{ @Override publicListparse(Readerr){..} } //解析json @Component publicclassJSONParserimplementsParser{ @Override publicListparse(Readerr){..} }
4、最后寫一個調用的客戶端,通過switch case根據不同的類型調用不同的實現
@Service publicclassClient{ privateParsercsvParser,jsonParser; @Autowired publicClient(ParsercsvParser,ParserjsonParser){ this.csvParser=csvParser; this.jsonParser=jsonParser; } publicListgetAll(ContentTypecontentType){ .. switch(contentType){ caseCSV: returncsvParser.parse(reader); caseJSON: returnjsonParser.parse(reader); .. } } .. }
可能大部分人都是像上面一樣的方式實現的,也能正常運行,那深入思考下,存在什么問題嗎?
現在假如產品經理提出了一個新需求要支持XML類型的文件,是不是客戶端也要修改代碼,需要在switch case中添加新的類型,這就導致客戶端和不同的解析器緊密耦合。
那么有什么更好的方法呢?
應用Service Locator Pattern
沒錯,那就是用上我們的服務定位模式Service Locator Pattern。
1、讓我們定義我們的服務定位器接口ParserFactory, 它有一個接受內容類型參數并返回Parser的方法。
publicinterfaceParserFactory{ ParsergetParser(ContentTypecontentType); }
2、我們配置ServiceLocatorFactoryBean使用ParserFactory作為服務定位器接口,ParserFactory這個接口不需要寫實現類。
@Configuration publicclassParserConfig{ @Bean("parserFactory") publicFactoryBeanserviceLocatorFactoryBean(){ ServiceLocatorFactoryBeanfactoryBean=newServiceLocatorFactoryBean(); //設置服務定位接口 factoryBean.setServiceLocatorInterface(ParserFactory.class); returnfactoryBean; } }
3、設置解析器Bean的名稱為類型名稱,方便服務定位
//設置bean的名稱和類型一致 @Component("CSV") publicclassCSVParserimplementsParser{..} @Component("JSON") publicclassJSONParserimplementsParser{..} @Component("XML") publicclassXMLParserimplementsParser{..}
4、修改枚舉, 添加XML
publicenumContentType{ JSON, CSV, XML }
5、最后用客戶端調用,直接根據類型調用對應的解析器,沒有了switch case
@Service publicclassClient{ privateParserFactoryparserFactory; @Autowired publicClient(ParserFactoryparserFactory){ this.parserFactory=parserFactory; } publicListgetAll(ContentTypecontentType){ .. //關鍵點,直接根據類型獲取 returnparserFactory .getParser(contentType) .parse(reader); } .. }
嘿嘿,我們已經成功地實現了我們的目標。現在再加新的類型,我們只要擴展添加新的解析器就行,再也不用修改客戶端了,滿足開閉原則。
如果你覺得Bean的名稱直接使用類型怪怪的,這邊可以建議你按照下面的方式來。
publicenumContentType{ JSON(TypeConstants.JSON_PARSER), CSV(TypeConstants.CSV_PARSER), XML(TypeConstants.XML_PARSER); privatefinalStringparserName; ContentType(StringparserName){ this.parserName=parserName; } @Override publicStringtoString(){ returnthis.parserName; } publicinterfaceTypeConstants{ StringCSV_PARSER="csvParser"; StringJSON_PARSER="jsonParser"; StringXML_PARSER="xmlParser"; } } @Component(TypeConstants.CSV_PARSER) publicclassCSVParserimplementsParser{..} @Component(TypeConstants.JSON_PARSER) publicclassJSONParserimplementsParser{..} @Component(TypeConstants.XML_PARSER) publicclassXMLParserimplementsParser{..}
剖析Service Locator Pattern
通過前面的例子,想必大家基本知道服務定位器模式如何使用了吧,現在我們深入剖析下。
服務定位器模式消除了客戶端對具體實現的依賴。以下引自Martin Fowler的文章總結了核心思想:“服務定位器背后的基本思想是擁有一個知道如何獲取應用程序可能需要的所有服務的對象。因此,此應用程序的服務定位器將有一個在需要時返回“服務”的方法。”
Spring的ServiceLocatorFactoryBean實現了FactoryBean接口,創建了Service Factory服務工廠Bean。
總結
我們通過使用服務定位器模式實現了一種擴展 Spring 控制反轉的絕妙方法。它幫助我們解決了依賴注入未提供最佳解決方案的用例。也就是說,依賴注入仍然是首選,并且在大多數情況下不應使用服務定位器來替代依賴注入。
審核編輯:湯梓紅
-
接口
+關注
關注
33文章
8691瀏覽量
151911 -
文件
+關注
關注
1文章
570瀏覽量
24821 -
spring
+關注
關注
0文章
340瀏覽量
14388 -
JSON
+關注
關注
0文章
119瀏覽量
7011
原文標題:Spring項目中用了這種模式,經理對我刮目相看
文章出處:【微信號:芋道源碼,微信公眾號:芋道源碼】歡迎添加關注!文章轉載請注明出處。
發布評論請先 登錄
相關推薦
評論