0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

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

jf_ro2CN3Fa ? 來源:JAVA旭陽 ? 2024-01-10 09:41 ? 次閱讀

來源:JAVA旭陽

不知道大家在項目中有沒有遇到過這樣的場景,根據(jù)傳入的類型,調(diào)用接口不同的實(shí)現(xiàn)類或者說服務(wù),比如根據(jù)文件的類型使用 CSV解析器或者JSON解析器,在調(diào)用的客戶端一般都是用if else去做判斷,比如類型等于JSON,我就用JSON解析器,那如果新加一個類型的解析器,是不是調(diào)用的客戶端還要修改呢?這顯然太耦合了,本文就介紹一種方法,服務(wù)定位模式Service Locator Pattern來解決,它幫助我們消除緊耦合實(shí)現(xiàn)及其依賴性,并提出將服務(wù)與其具體類解耦。

文件解析器的例子

我們通過一個例子來告訴你如何使用Service Locator Pattern。

假設(shè)我們有一個從各種來源獲取數(shù)據(jù)的應(yīng)用程序,我們必須解析不同類型的文件,比如解析CSV文件和JSON文件。

1、定義一個類型的枚舉

publicenumContentType{
JSON,
CSV
}

2、定義一個解析的接口

publicinterfaceParser{
Listparse(Readerr);
}

3、根據(jù)不同的文件類型有不同的實(shí)現(xiàn)類

//解析csv
@Component
publicclassCSVParserimplementsParser{
@Override
publicListparse(Readerr){..}
}

//解析json
@Component
publicclassJSONParserimplementsParser{
@Override
publicListparse(Readerr){..}
}

4、最后寫一個調(diào)用的客戶端,通過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)行,那深入思考下,存在什么問題嗎?

現(xiàn)在假如產(chǎn)品經(jīng)理提出了一個新需求要支持XML類型的文件,是不是客戶端也要修改代碼,需要在switch case中添加新的類型,這就導(dǎo)致客戶端和不同的解析器緊密耦合。

那么有什么更好的方法呢?

應(yīng)用Service Locator Pattern

沒錯,那就是用上我們的服務(wù)定位模式Service Locator Pattern。

1、讓我們定義我們的服務(wù)定位器接口ParserFactory, 它有一個接受內(nèi)容類型參數(shù)并返回Parser的方法。

publicinterfaceParserFactory{
ParsergetParser(ContentTypecontentType);
}

2、我們配置ServiceLocatorFactoryBean使用ParserFactory作為服務(wù)定位器接口,ParserFactory這個接口不需要寫實(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、修改枚舉, 添加XML

publicenumContentType{
JSON,
CSV,
XML
}

5、最后用客戶端調(diào)用,直接根據(jù)類型調(diào)用對應(yīng)的解析器,沒有了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ò)展添加新的解析器就行,再也不用修改客戶端了,滿足開閉原則。

如果你覺得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

通過前面的例子,想必大家基本知道服務(wù)定位器模式如何使用了吧,現(xiàn)在我們深入剖析下。

服務(wù)定位器模式消除了客戶端對具體實(shí)現(xiàn)的依賴。以下引自Martin Fowler的文章總結(jié)了核心思想:“服務(wù)定位器背后的基本思想是擁有一個知道如何獲取應(yīng)用程序可能需要的所有服務(wù)的對象。因此,此應(yīng)用程序的服務(wù)定位器將有一個在需要時返回“服務(wù)”的方法。”

8e123e42-af58-11ee-8b88-92fbcf53809c.png

Spring的ServiceLocatorFactoryBean實(shí)現(xiàn)了FactoryBean接口,創(chuàng)建了Service Factory服務(wù)工廠Bean。

總結(jié)

我們通過使用服務(wù)定位器模式實(shí)現(xiàn)了一種擴(kuò)展 Spring 控制反轉(zhuǎn)的絕妙方法。它幫助我們解決了依賴注入未提供最佳解決方案的用例。也就是說,依賴注入仍然是首選,并且在大多數(shù)情況下不應(yīng)使用服務(wù)定位器來替代依賴注入。

審核編輯:湯梓紅

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報投訴
  • 接口
    +關(guān)注

    關(guān)注

    33

    文章

    8352

    瀏覽量

    150508
  • 文件
    +關(guān)注

    關(guān)注

    1

    文章

    555

    瀏覽量

    24633
  • spring
    +關(guān)注

    關(guān)注

    0

    文章

    335

    瀏覽量

    14277
  • JSON
    +關(guān)注

    關(guān)注

    0

    文章

    114

    瀏覽量

    6919

原文標(biāo)題:Spring項目中用了這種模式,經(jīng)理對我刮目相看

文章出處:【微信號:芋道源碼,微信公眾號:芋道源碼】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。

收藏 人收藏

    評論

    相關(guān)推薦

    java spring教程

    Spring核心概念介紹控制反轉(zhuǎn)(IOC)依賴注入(DI)集合對象注入等Bean的管理BeanFactoryApplicationContextSpring 在web中的使用
    發(fā)表于 09-11 11:09

    什么是java spring

    通過一種稱作控制反轉(zhuǎn)(IoC)的技術(shù)促進(jìn)了松耦合。當(dāng)應(yīng)用了IoC,個對象依賴的其它對象會通過被動的方式傳遞進(jìn)來,而不是這個對象自己創(chuàng)建或者查找依賴對象。你可以認(rèn)為IoC與JNDI相反
    發(fā)表于 09-11 11:16

    Spring工作原理

    核心就是AOP這個就是面向切面編程,可以為某類對象 進(jìn)行監(jiān)督和控制(也就是在調(diào)用這類對象的具體方法的前后去調(diào)用你指定的 模塊)從而達(dá)到對個模塊擴(kuò)充的功能。這些都是通過配置類達(dá)到的。
    發(fā)表于 07-10 07:41

    Spring筆記分享

    框架:高度抽取,可重用代碼的一種設(shè)計高度的通用性多個可重用模塊的集合,形成某個領(lǐng)域的整體解決方案Spring => 容器框架包含并管理應(yīng)用對象的生命周期IOC和SOP容器框架容器 =>
    發(fā)表于 11-04 07:51

    啟動Spring Boot項目應(yīng)用的三種方法

    分別對這三啟動方法做簡單的介紹。第一種、IDEA編輯器啟動。在我們開發(fā)過程中,用IDEA編輯器啟動的方式最常見最推薦的啟動方式。我們在項目中找到我們的啟動文件,鼠標(biāo)右擊run啟動文件即可。如下圖所示
    發(fā)表于 01-14 17:33

    請問如何去實(shí)現(xiàn)一種按鍵控制電機(jī)的正反轉(zhuǎn)?

    如何去實(shí)現(xiàn)一種按鍵控制電機(jī)的正反轉(zhuǎn)?按鍵控制電機(jī)的功能是什么?按鍵控制電機(jī)的接線方法有哪些?
    發(fā)表于 06-29 07:09

    spring教程ppt

    主要內(nèi)容Spring 概述Spring 整體結(jié)構(gòu)Spring實(shí)例Spring核心概念介紹控制反轉(zhuǎn)
    發(fā)表于 09-11 11:00 ?138次下載
    <b class='flag-5'>spring</b>教程ppt

    一種金融系統(tǒng)專用ETL工具的研究與實(shí)現(xiàn)

    實(shí)現(xiàn)了一種基于Spring框架的商業(yè)銀行專用ETL程序。該程序利用Spring框架的控制反轉(zhuǎn)技術(shù),基于Ibatis的數(shù)據(jù)訪問對象技術(shù)和
    發(fā)表于 04-13 09:04 ?24次下載

    一種防止相間短路的電動機(jī)正反轉(zhuǎn)控制

    一種防止相間短路的電動機(jī)正反轉(zhuǎn)控制
    發(fā)表于 10-09 22:30 ?531次閱讀
    另<b class='flag-5'>一種</b>防止相間短路的電動機(jī)正<b class='flag-5'>反轉(zhuǎn)</b><b class='flag-5'>控制</b>

    一種防止相間短路的正反轉(zhuǎn)控制電路圖

    一種防止相間短路的正反轉(zhuǎn)控制
    發(fā)表于 03-03 12:49 ?1017次閱讀
    另<b class='flag-5'>一種</b>防止相間短路的正<b class='flag-5'>反轉(zhuǎn)</b><b class='flag-5'>控制</b>電路圖

    Spring認(rèn)證_什么是Spring GraphQL?

    數(shù)據(jù)整合 Spring GraphQL 支持使用 Querydsl 通過 Spring Data Querydsl 擴(kuò)展 來獲取數(shù)據(jù)。Querydsl 提供了一種靈活但類型安全的
    的頭像 發(fā)表于 08-09 11:31 ?563次閱讀
    <b class='flag-5'>Spring</b>認(rèn)證_什么是<b class='flag-5'>Spring</b> GraphQL?

    剖析Spring中最常用的擴(kuò)展點(diǎn)(上)

    我們說到spring,可能第個想到的是 `IOC`(控制反轉(zhuǎn)) 和 `AOP`(面向切面編程)。 沒錯,它們是
    的頭像 發(fā)表于 02-15 16:06 ?699次閱讀
    剖析<b class='flag-5'>Spring</b>中最常用的<b class='flag-5'>擴(kuò)展</b>點(diǎn)(上)

    剖析Spring中最常用的擴(kuò)展點(diǎn)(中)

    我們說到spring,可能第個想到的是 `IOC`(控制反轉(zhuǎn)) 和 `AOP`(面向切面編程)。 沒錯,它們是
    的頭像 發(fā)表于 02-15 16:06 ?457次閱讀
    剖析<b class='flag-5'>Spring</b>中最常用的<b class='flag-5'>擴(kuò)展</b>點(diǎn)(中)

    剖析Spring中最常用的擴(kuò)展點(diǎn)(下)

    我們說到spring,可能第個想到的是 `IOC`(控制反轉(zhuǎn)) 和 `AOP`(面向切面編程)。 沒錯,它們是
    的頭像 發(fā)表于 02-15 16:07 ?415次閱讀

    基于spring的SPI擴(kuò)展機(jī)制是如何實(shí)現(xiàn)的?

    基本上,你說是基于 spring 的 SPI 擴(kuò)展機(jī)制,再把spring.factories文件和EnableAutoConfiguration提
    的頭像 發(fā)表于 03-07 09:17 ?968次閱讀