來(lái)源:JAVA旭陽(yáng)
不知道大家在項(xiàng)目中有沒(méi)有遇到過(guò)這樣的場(chǎng)景,根據(jù)傳入的類型,調(diào)用接口不同的實(shí)現(xiàn)類或者說(shuō)服務(wù),比如根據(jù)文件的類型使用 CSV解析器或者JSON解析器,在調(diào)用的客戶端一般都是用if else去做判斷,比如類型等于JSON,我就用JSON解析器,那如果新加一個(gè)類型的解析器,是不是調(diào)用的客戶端還要修改呢?這顯然太耦合了,本文就介紹一種方法,服務(wù)定位模式Service Locator Pattern來(lái)解決,它幫助我們消除緊耦合實(shí)現(xiàn)及其依賴性,并提出將服務(wù)與其具體類解耦。
文件解析器的例子
我們通過(guò)一個(gè)例子來(lái)告訴你如何使用Service Locator Pattern。
假設(shè)我們有一個(gè)從各種來(lái)源獲取數(shù)據(jù)的應(yīng)用程序,我們必須解析不同類型的文件,比如解析CSV文件和JSON文件。
1、定義一個(gè)類型的枚舉
publicenumContentType{
JSON,
CSV
}
2、定義一個(gè)解析的接口
publicinterfaceParser{ Listparse(Readerr); }
3、根據(jù)不同的文件類型有不同的實(shí)現(xiàn)類
//解析csv
@Component
publicclassCSVParserimplementsParser{
@Override
publicListparse(Readerr){..}
}
//解析json
@Component
publicclassJSONParserimplementsParser{
@Override
publicListparse(Readerr){..}
}
4、最后寫一個(gè)調(diào)用的客戶端,通過(guò)switch case根據(jù)不同的類型調(diào)用不同的實(shí)現(xiàn)
@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);
..
}
}
..
}
可能大部分人都是像上面一樣的方式實(shí)現(xiàn)的,也能正常運(yùn)行,那深入思考下,存在什么問(wèn)題嗎?
現(xiàn)在假如產(chǎn)品經(jīng)理提出了一個(gè)新需求要支持XML類型的文件,是不是客戶端也要修改代碼,需要在switch case中添加新的類型,這就導(dǎo)致客戶端和不同的解析器緊密耦合。
那么有什么更好的方法呢?
應(yīng)用Service Locator Pattern
沒(méi)錯(cuò),那就是用上我們的服務(wù)定位模式Service Locator Pattern。
1、讓我們定義我們的服務(wù)定位器接口ParserFactory, 它有一個(gè)接受內(nèi)容類型參數(shù)并返回Parser的方法。
publicinterfaceParserFactory{
ParsergetParser(ContentTypecontentType);
}
2、我們配置ServiceLocatorFactoryBean使用ParserFactory作為服務(wù)定位器接口,ParserFactory這個(gè)接口不需要寫實(shí)現(xiàn)類。
@Configuration publicclassParserConfig{ @Bean("parserFactory") publicFactoryBeanserviceLocatorFactoryBean(){ ServiceLocatorFactoryBeanfactoryBean=newServiceLocatorFactoryBean(); //設(shè)置服務(wù)定位接口 factoryBean.setServiceLocatorInterface(ParserFactory.class); returnfactoryBean; } }
3、設(shè)置解析器Bean的名稱為類型名稱,方便服務(wù)定位
//設(shè)置bean的名稱和類型一致
@Component("CSV")
publicclassCSVParserimplementsParser{..}
@Component("JSON")
publicclassJSONParserimplementsParser{..}
@Component("XML")
publicclassXMLParserimplementsParser{..}
4、修改枚舉, 添加X(jué)ML
publicenumContentType{
JSON,
CSV,
XML
}
5、最后用客戶端調(diào)用,直接根據(jù)類型調(diào)用對(duì)應(yīng)的解析器,沒(méi)有了switch case
@Service
publicclassClient{
privateParserFactoryparserFactory;
@Autowired
publicClient(ParserFactoryparserFactory){
this.parserFactory=parserFactory;
}
publicListgetAll(ContentTypecontentType){
..
//關(guān)鍵點(diǎn),直接根據(jù)類型獲取
returnparserFactory
.getParser(contentType)
.parse(reader);
}
..
}
嘿嘿,我們已經(jīng)成功地實(shí)現(xiàn)了我們的目標(biāo)?,F(xiàn)在再加新的類型,我們只要擴(kuò)展添加新的解析器就行,再也不用修改客戶端了,滿足開閉原則。
如果你覺(jué)得Bean的名稱直接使用類型怪怪的,這邊可以建議你按照下面的方式來(lái)。
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
通過(guò)前面的例子,想必大家基本知道服務(wù)定位器模式如何使用了吧,現(xiàn)在我們深入剖析下。
服務(wù)定位器模式消除了客戶端對(duì)具體實(shí)現(xiàn)的依賴。以下引自Martin Fowler的文章總結(jié)了核心思想:“服務(wù)定位器背后的基本思想是擁有一個(gè)知道如何獲取應(yīng)用程序可能需要的所有服務(wù)的對(duì)象。因此,此應(yīng)用程序的服務(wù)定位器將有一個(gè)在需要時(shí)返回“服務(wù)”的方法。”

Spring的ServiceLocatorFactoryBean實(shí)現(xiàn)了FactoryBean接口,創(chuàng)建了Service Factory服務(wù)工廠Bean。
總結(jié)
我們通過(guò)使用服務(wù)定位器模式實(shí)現(xiàn)了一種擴(kuò)展 Spring 控制反轉(zhuǎn)的絕妙方法。它幫助我們解決了依賴注入未提供最佳解決方案的用例。也就是說(shuō),依賴注入仍然是首選,并且在大多數(shù)情況下不應(yīng)使用服務(wù)定位器來(lái)替代依賴注入。
審核編輯:湯梓紅
-
接口
+關(guān)注
關(guān)注
33文章
9306瀏覽量
155678 -
文件
+關(guān)注
關(guān)注
1文章
586瀏覽量
25867 -
spring
+關(guān)注
關(guān)注
0文章
341瀏覽量
15661 -
JSON
+關(guān)注
關(guān)注
0文章
122瀏覽量
7665
原文標(biāo)題:Spring項(xiàng)目中用了這種模式,經(jīng)理對(duì)我刮目相看
文章出處:【微信號(hào):芋道源碼,微信公眾號(hào):芋道源碼】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。
發(fā)布評(píng)論請(qǐng)先 登錄
java spring教程
什么是java spring
Spring工作原理
Spring筆記分享
啟動(dòng)Spring Boot項(xiàng)目應(yīng)用的三種方法
請(qǐng)問(wèn)如何去實(shí)現(xiàn)一種按鍵控制電機(jī)的正反轉(zhuǎn)?
spring教程ppt
一種金融系統(tǒng)專用ETL工具的研究與實(shí)現(xiàn)
Spring認(rèn)證_什么是Spring GraphQL?
剖析Spring中最常用的擴(kuò)展點(diǎn)(上)
剖析Spring中最常用的擴(kuò)展點(diǎn)(中)

一種擴(kuò)展Spring控制反轉(zhuǎn)的絕妙方法
評(píng)論