成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

Spring項目中用了這種模式,經理對我刮目相看

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

?前言

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

文件解析器的例子

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

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

  1. 定義一個類型的枚舉
public enum ContentType {
JSON,
CSV
}
  1. 定義一個解析的接口
public interface Parser {
List parse(Reader r);
}
  1. 根據不同的文件類型有不同的實現(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. 最后寫一個調用的客戶端,通過switch case根據不同的類型調用不同的實現(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);
..
}
}
..
}

可能大部分人都是像上面一樣的方式實現(xiàn)的,也能正常運行,那深入思考下,存在什么問題嗎?

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

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

應用Service Locator Pattern

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

  1. 讓我們定義我們的服務定位器接口ParserFactory, 它有一個接受內容類型參數(shù)并返回Parser的方法。
public interface ParserFactory {
Parser getParser(ContentType contentType);
}
  1. 我們配置ServiceLocatorFactoryBean使用ParserFactory作為服務定位器接口,ParserFactory這個接口不需要寫實現(xiàn)類。
@Configuration
public class ParserConfig {

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

}
  1. 設置解析器Bean的名稱為類型名稱,方便服務定位
// 設置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. 最后用客戶端調用,直接根據類型調用對應的解析器,沒有了switch case
@Service
public class Client {
private ParserFactory parserFactory;
@Autowired
public Client(ParserFactory parserFactory) {
this.parserFactory = parserFactory;
}
public List getAll(ContentType contentType) {
..
// 關鍵點,直接根據類型獲取
return parserFactory
.getParser(contentType)
.parse(reader);
}
..
}

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

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

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

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

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

圖片

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

總結

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

責任編輯:武曉燕 來源: JAVA旭陽
相關推薦

2019-08-23 09:20:35

Spring 5編程Java

2024-03-15 13:06:00

代碼技巧ES6-

2015-11-02 10:00:31

數(shù)據格局

2022-09-21 09:01:27

Spring設計模式框架,

2021-11-19 15:37:14

大數(shù)據初創(chuàng)公司盤點

2019-05-14 08:44:13

面試面試官線程安全

2010-10-20 09:37:00

vi編輯器

2018-11-07 09:39:03

Runtime開發(fā)項目

2025-02-24 10:36:15

2021-01-22 19:13:12

騰訊云

2020-08-31 08:14:46

Python開發(fā)代碼

2017-05-24 10:12:54

前端FlexboxCSS3

2025-03-28 08:53:51

2024-05-30 07:37:30

2010-05-17 17:49:54

2013-10-10 09:23:13

項目經理項目

2024-12-10 07:10:00

2024-09-09 05:30:00

數(shù)據庫Spring

2021-03-01 12:51:00

監(jiān)控產品經理指標

2021-05-06 09:06:12

Vue Router組件視圖
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 麻豆av网站 | 黄网站免费在线看 | 一级毛片大全免费播放 | 久久欧美精品 | 91精品国产综合久久久密闭 | 亚洲精品视频一区二区三区 | 欧美日韩精品久久久免费观看 | 97天天干 | 国产综合久久 | 亚洲一区高清 | 国产一二三视频在线观看 | 91麻豆精品国产91久久久更新资源速度超快 | 精品国产aⅴ | 久久99精品久久 | 国产精品久久久久久久久久免费看 | 久久伊人精品一区二区三区 | 亚洲国产精品久久久久婷婷老年 | 992人人草| 成人福利在线 | 精品1区2区3区4区 | 九九九久久国产免费 | 97久久精品午夜一区二区 | 天天干狠狠操 | 九九精品在线 | 成年人视频免费在线观看 | 中文字幕av在线 | 青青操av| 亚洲免费在线观看 | 精品麻豆剧传媒av国产九九九 | 国产一区二区三区四区区 | 国内自拍第一页 | 亚洲成人av | 一区二区三区视频在线观看 | 91久久精品一区二区二区 | 精品在线一区二区三区 | 在线视频成人 | 久久久久久国产精品免费免费 | 自拍偷拍在线视频 | 免费看a| 亚洲精品视频二区 | 中文字幕在线第一页 |