欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Spring你不知道的一種解耦模式

 更新時(shí)間:2023年01月19日 11:24:35   作者:Java知識(shí)庫(kù)  
本文介紹一種方法,服務(wù)定位模式Service?Locator?Pattern來(lái)解決,它幫助我們消除緊耦合實(shí)現(xiàn)及其依賴(lài)性,并提出將服務(wù)與其具體類(lèi)解耦

前言

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

一個(gè)例子入門(mén)

我們通過(guò)一個(gè)例子來(lái)告訴你如何使用Service Locator Pattern。

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

定義一個(gè)類(lèi)型的枚舉

public enum ContentType {
  JSON,
  CSV
}

定義一個(gè)解析的接口

public interface Parser {
  List parse(Reader r);
}

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

// 解析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) { .. }
}

最后寫(xiě)一個(gè)調(diào)用的客戶(hù)端,通過(guò)switch case根據(jù)不同的類(lèi)型調(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類(lèi)型的文件,是不是客戶(hù)端也要修改代碼,需要在switch case中添加新的類(lèi)型,這就導(dǎo)致客戶(hù)端和不同的解析器緊密耦合。

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

應(yīng)用Service Locator Pattern

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

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

public interface ParserFactory {
  Parser getParser(ContentType contentType);
}

我們配置ServiceLocatorFactoryBean使用ParserFactory作為服務(wù)定 位器接口,ParserFactory這個(gè)接口不需要寫(xiě)實(shí)現(xiàn)類(lèi)。

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

設(shè)置解析器Bean的名稱(chēng)為類(lèi)型名稱(chēng),方便服務(wù)定位

// 設(shè)置bean的名稱(chēng)和類(lèi)型一致
@Component("CSV")
public class CSVParser implements Parser { .. }
@Component("JSON")
public class JSONParser implements Parser { .. }
@Component("XML")
public class XMLParser implements Parser { .. }

修改枚舉, 添加X(jué)ML

public enum ContentType {
  JSON,
  CSV,
  XML
}

最后用客戶(hù)端調(diào)用,直接根據(jù)類(lèi)型調(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ù)類(lèi)型獲取
    return parserFactory
        .getParser(contentType)  
        .parse(reader);
  }
  ..
}

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

如果你覺(jué)得Bean的名稱(chēng)直接使用類(lèi)型怪怪的,這邊可以建議你按照下面的方式來(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ù)定 位器模式消除了客戶(hù)端對(duì)具體實(shí)現(xiàn)的依賴(lài)。以下引自 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)的絕妙方法。它幫助我們解決了依賴(lài)注入未提供最佳解決方案的用例。也就是說(shuō),依賴(lài)注入仍然是首選,并且在大多數(shù)情況下不應(yīng)使用服務(wù)定 位器來(lái)替代依賴(lài)注入。

到此這篇關(guān)于Spring你不知道的一種解耦模式的文章就介紹到這了,更多相關(guān)Spring解耦模式內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

最新評(píng)論