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

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

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

Spring項(xiàng)目中用這種模式更方便

jf_78858299 ? 來(lái)源:JAVA旭陽(yáng) ? 作者:JAVA旭陽(yáng) ? 2023-05-11 10:39 ? 次閱讀

前言

不知道大家在項(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è)類型的枚舉
public enum ContentType {
  JSON,
  CSV
}
  1. 定義一個(gè)解析的接口
public interface Parser {
  List parse(Reader r);
}
  1. 根據(jù)不同的文件類型有不同的實(shí)現(xiàn)類
// 解析csv
@Component
public class CSVParser implements Parser { 
  @Override
  public List parse(Reader r) { .. }
}

// 解析json
@Component
public class JSONParser implements Parser {
  @Override
  public List parse(Reader r) { .. }
}
  1. 最后寫一個(gè)調(diào)用的客戶端,通過(guò)switch case根據(jù)不同的類型調(diào)用不同的實(shí)現(xiàn)
@Service
public class Client {
  private Parser csvParser, jsonParser;

  @Autowired
  public Client(Parser csvParser, Parser jsonParser) {
    this.csvParser = csvParser;
    this.jsonParser = jsonParser;
  }

  public List getAll(ContentType contentType) {
    ..

    switch (contentType) {
      case CSV:
        return csvParser.parse(reader);
      case JSON:
        return jsonParser.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的方法。
public interface ParserFactory {
  Parser getParser(ContentType contentType);
}
  1. 我們配置ServiceLocatorFactoryBean使用ParserFactory作為服務(wù)定位器接口,ParserFactory這個(gè)接口不需要寫實(shí)現(xiàn)類。
@Configuration
public class ParserConfig {

  @Bean("parserFactory")
  public FactoryBean serviceLocatorFactoryBean() {
    ServiceLocatorFactoryBean factoryBean = new ServiceLocatorFactoryBean();
    // 設(shè)置服務(wù)定位接口   
    factoryBean.setServiceLocatorInterface(ParserFactory.class);
    return factoryBean;
  }

}
  1. 設(shè)置解析器Bean的名稱為類型名稱,方便服務(wù)定位
// 設(shè)置bean的名稱和類型一致
@Component("CSV")
public class CSVParser implements Parser { .. }
@Component("JSON")
public class JSONParser implements Parser { .. }
@Component("XML")
public class XMLParser implements Parser { .. }
  1. 修改枚舉, 添加XML
public enum ContentType {
  JSON,
  CSV,
  XML
}
  1. 最后用客戶端調(diào)用,直接根據(jù)類型調(diào)用對(duì)應(yīng)的解析器,沒(méi)有了switch case
@Service
public class Client {
  private ParserFactory parserFactory;
  @Autowired
  public Client(ParserFactory parserFactory) {
    this.parserFactory = parserFactory;
  }
  public List getAll(ContentType contentType) {
    ..
    // 關(guān)鍵點(diǎn),直接根據(jù)類型獲取
    return parserFactory
        .getParser(contentType)  
        .parse(reader);
  }
  ..
}

嘿嘿,我們已經(jīng)成功地實(shí)現(xiàn)了我們的目標(biāo)?,F(xiàn)在再加新的類型,我們只要擴(kuò)展添加新的解析器就行,再也不用修改客戶端了,滿足開閉原則。

如果你覺得Bean的名稱直接使用類型怪怪的,這邊可以建議你按照下面的方式來(lái)。

public enum ContentType {
  JSON(TypeConstants.JSON_PARSER),
  CSV(TypeConstants.CSV_PARSER),
  XML(TypeConstants.XML_PARSER);
  private final String parserName;
  ContentType(String parserName) {
    this.parserName = parserName;
  }

  @Override
  public String toString() {
    return this.parserName;
  }
  public interface TypeConstants {

    String CSV_PARSER = "csvParser";
    String JSON_PARSER = "jsonParser";
    String XML_PARSER = "xmlParser"; 
  }
}

@Component(TypeConstants.CSV_PARSER)
public class CSVParser implements Parser { .. }
@Component(TypeConstants.JSON_PARSER)
public class JSONParser implements Parser { .. }
@Component(TypeConstants.XML_PARSER)
public class XMLParser implements Parser { .. }

剖析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ù)”的方法。”

圖片

SpringServiceLocatorFactoryBean實(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)替代依賴注入。

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

    關(guān)注

    33

    文章

    8448

    瀏覽量

    150725
  • JSON
    +關(guān)注

    關(guān)注

    0

    文章

    116

    瀏覽量

    6929
  • csv
    csv
    +關(guān)注

    關(guān)注

    0

    文章

    38

    瀏覽量

    5797
收藏 人收藏

    評(píng)論

    相關(guān)推薦

    Gradle構(gòu)建的多模塊Spring Boot項(xiàng)目

    構(gòu)建的多模塊 Spring Boot 項(xiàng)目,JDK 版本為8,IDEA 版本為 2022.3.1 。 1. 創(chuàng)建項(xiàng)目 打開IDEA,選擇菜單:File -> New -> Project,在其
    的頭像 發(fā)表于 09-25 14:46 ?2803次閱讀
    Gradle構(gòu)建的多模塊<b class='flag-5'>Spring</b> Boot<b class='flag-5'>項(xiàng)目</b>

    java spring教程

    java spring教程理解Spring 實(shí)現(xiàn)原理掌握Spring IOC,AOP掌握Spring的基礎(chǔ)配置和用法熟練使用SSH開發(fā)項(xiàng)目
    發(fā)表于 09-11 11:09

    什么是java spring

    。在SSH項(xiàng)目中管理事務(wù)以及對(duì)象的注入Spring是非侵入式的:基于Spring開發(fā)的系統(tǒng)中的對(duì)象一般不依賴于Spring的類。組成 Spring
    發(fā)表于 09-11 11:16

    如何把自己做的項(xiàng)目中用到的所有子vi打包成庫(kù)類?

    如題,如何將自己做的項(xiàng)目中的所有子vi打包成庫(kù)類,方便以后用到的時(shí)候調(diào)用?因?yàn)楝F(xiàn)在做了挺多的項(xiàng)目,其中有很多都是用的寫好的功能的子vi,每次寫新的項(xiàng)目的時(shí)候都要重新復(fù)制到新的
    發(fā)表于 11-23 16:33

    Spring MVC練手項(xiàng)目

    初識(shí) Spring MVC——練手小項(xiàng)目
    發(fā)表于 09-17 08:41

    如何在我的項(xiàng)目中使用停止模式?

    你好,我想在我的項(xiàng)目中使用停止模式。有什么例子嗎?我想讓我的外圍模塊在初始化時(shí)停止模式。如果用戶將喚醒按鈕,模塊醒來(lái)并開始廣告。模塊進(jìn)入停止模式,再然后preconfiguredtim
    發(fā)表于 09-25 14:58

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

    、方便。打個(gè)比方,如果我們做傳統(tǒng)的spring web項(xiàng)目,我們需要做哪些工作。1)配置web.xml,加載springspring mv
    發(fā)表于 01-14 17:33

    Spring認(rèn)證」Spring Hello World 項(xiàng)目示例

    。現(xiàn)在使用向?qū)Т翱趯⒛?b class='flag-5'>項(xiàng)目命名為HelloSpring,如下所示 -成功創(chuàng)建項(xiàng)目后,您的項(xiàng)目資源管理器中將包含以下內(nèi)容-第 2 步 - 添加所需的庫(kù)第二步,讓我們?cè)?b class='flag-5'>項(xiàng)目中添加
    發(fā)表于 08-17 13:49

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

    Spring GraphQL 為構(gòu)建在 GraphQL Java 上的 Spring 應(yīng)用程序提供支持。兩個(gè)團(tuán)隊(duì)之間的聯(lián)合聯(lián)合。我們的共同理念是少固執(zhí)己見,專注于全面和廣泛的支持。 Spri
    的頭像 發(fā)表于 08-06 14:30 ?678次閱讀
    <b class='flag-5'>Spring</b>認(rèn)證_什么是<b class='flag-5'>Spring</b> GraphQL

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

    這個(gè)項(xiàng)目建立在 Boot 2.x 上,但它應(yīng)該與最新的 Boot2.4.x5 相關(guān)。 要?jiǎng)?chuàng)建項(xiàng)目,請(qǐng)轉(zhuǎn)到start.spring.io并為要使用的GraphQL傳輸選擇啟動(dòng)器: 啟動(dòng)機(jī) 運(yùn)輸 執(zhí)行
    的頭像 發(fā)表于 08-10 14:08 ?788次閱讀
    「<b class='flag-5'>Spring</b>認(rèn)證」什么是<b class='flag-5'>Spring</b> GraphQL?

    機(jī)器視覺項(xiàng)目中需用到的光源

    不同效果 。 根據(jù)我們的統(tǒng)計(jì),機(jī)器視覺項(xiàng)目中用到的光源有如下幾種: 環(huán)形光源(LQ-HDRmmnn-C): 光出射角度值在0°~90°。0°~45°為低角度環(huán)形光源,目前應(yīng)用案例包括手機(jī)金屬外框劃痕檢測(cè)、光滑表面的劃痕、破損檢測(cè)以達(dá)到突顯物體輪廓
    的頭像 發(fā)表于 05-30 09:26 ?872次閱讀

    Spring中用到了哪些設(shè)計(jì)模式

    Spring 通過(guò)動(dòng)態(tài)代理對(duì)類進(jìn)行方法級(jí)別的切面增強(qiáng),動(dòng)態(tài)生成目標(biāo)對(duì)象的代理類,并在代理類的方法中設(shè)置攔截器,通過(guò)執(zhí)行攔截器中的邏輯增強(qiáng)了代理方法的功能,從而實(shí)現(xiàn) AOP。
    發(fā)表于 05-30 09:43 ?286次閱讀
    <b class='flag-5'>Spring</b><b class='flag-5'>中用</b>到了哪些設(shè)計(jì)<b class='flag-5'>模式</b>

    kafka client在 spring如何實(shí)現(xiàn)

    之前寫過(guò)關(guān)于 Apache Pulsar 的簡(jiǎn)單示例,用來(lái)了解如何使用 Pulsar 這個(gè)新生代的消息隊(duì)列中間件,但是如果想要在項(xiàng)目中使用,還會(huì)欠缺很多,最明顯的就是 集成復(fù)雜,如果你用過(guò)其他
    的頭像 發(fā)表于 09-25 11:21 ?452次閱讀
    kafka client在 <b class='flag-5'>spring</b>如何實(shí)現(xiàn)

    Spring Boot的啟動(dòng)原理

    spring-boot-maven-plugin 的 maven 項(xiàng)目打包插件,可以方便的將 Spring Boot 項(xiàng)目打成 jar 包
    的頭像 發(fā)表于 10-13 11:44 ?598次閱讀
    <b class='flag-5'>Spring</b> Boot的啟動(dòng)原理

    Spring中經(jīng)典的9種設(shè)計(jì)模式

    spring中常用的設(shè)計(jì)模式達(dá)到九種,我們一一舉例
    的頭像 發(fā)表于 12-11 09:56 ?904次閱讀
    <b class='flag-5'>Spring</b>中經(jīng)典的9種設(shè)計(jì)<b class='flag-5'>模式</b>