Redian新闻
>
Spring项目中用了这种模式,经理对我刮目相看

Spring项目中用了这种模式,经理对我刮目相看

公众号新闻

点击上方“芋道源码”,选择“设为星标

管她前浪,还是后浪?

能浪的浪,才是好浪!

每天 10:33 更新文章,每天掉亿点点头发...

源码精品专栏

 
来源:JAVA旭阳

不知道大家在项目中有没有遇到过这样的场景,根据传入的类型,调用接口不同的实现类或者说服务,比如根据文件的类型使用 CSV解析器或者JSON解析器,在调用的客户端一般都是用if else去做判断,比如类型等于JSON,我就用JSON解析器,那如果新加一个类型的解析器,是不是调用的客户端还要修改呢?这显然太耦合了,本文就介绍一种方法,服务定位模式Service Locator Pattern来解决,它帮助我们消除紧耦合实现及其依赖性,并提出将服务与其具体类解耦。

文件解析器的例子

我们通过一个例子来告诉你如何使用Service Locator Pattern

假设我们有一个从各种来源获取数据的应用程序,我们必须解析不同类型的文件,比如解析CSV文件和JSON文件。

1、定义一个类型的枚举

public enum ContentType {
  JSON,
  CSV
}

2、定义一个解析的接口

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

3、根据不同的文件类型有不同的实现类

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

4、最后写一个调用的客户端,通过switch case根据不同的类型调用不同的实现

@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);
      ..
    }
  }
  ..
}

可能大部分人都是像上面一样的方式实现的,也能正常运行,那深入思考下,存在什么问题吗?

现在假如产品经理提出了一个新需求要支持XML类型的文件,是不是客户端也要修改代码,需要在switch case中添加新的类型,这就导致客户端和不同的解析器紧密耦合。

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

基于 Spring Boot + MyBatis Plus + Vue & Element 实现的后台管理系统 + 用户小程序,支持 RBAC 动态权限、多租户、数据权限、工作流、三方登录、支付、短信、商城等功能

  • 项目地址:https://github.com/YunaiV/ruoyi-vue-pro
  • 视频教程:https://doc.iocoder.cn/video/

应用Service Locator Pattern

没错,那就是用上我们的服务定位模式Service Locator Pattern

1、让我们定义我们的服务定位器接口ParserFactory, 它有一个接受内容类型参数并返回Parser的方法。

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

2、我们配置ServiceLocatorFactoryBean使用ParserFactory作为服务定位器接口,ParserFactory这个接口不需要写实现类。

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

}

3、设置解析器Bean的名称为类型名称,方便服务定位

// 设置bean的名称和类型一致
@Component("CSV")
public class CSVParser implements Parser { .. }
@Component("JSON")
public class JSONParser implements Parser { .. }
@Component("XML")
public class XMLParser implements Parser { .. }

4、修改枚举, 添加XML

public enum ContentType {
  JSON,
  CSV,
  XML
}

5、最后用客户端调用,直接根据类型调用对应的解析器,没有了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);
  }
  ..
}

嘿嘿,我们已经成功地实现了我们的目标。现在再加新的类型,我们只要扩展添加新的解析器就行,再也不用修改客户端了,满足开闭原则。

如果你觉得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 { .. }

基于 Spring Cloud Alibaba + Gateway + Nacos + RocketMQ + Vue & Element 实现的后台管理系统 + 用户小程序,支持 RBAC 动态权限、多租户、数据权限、工作流、三方登录、支付、短信、商城等功能

  • 项目地址:https://github.com/YunaiV/yudao-cloud
  • 视频教程:https://doc.iocoder.cn/video/

剖析Service Locator Pattern

通过前面的例子,想必大家基本知道服务定位器模式如何使用了吧,现在我们深入剖析下。

服务定位器模式 消除了客户端对具体实现的依赖。以下引自 Martin Fowler 的文章总结了核心思想:“服务定位器背后的基本思想是拥有一个知道如何获取应用程序可能需要的所有服务的对象。因此,此应用程序的服务定位器将有一个在需要时返回“服务”的方法。”

SpringServiceLocatorFactoryBean实现了 FactoryBean接口,创建了Service Factory服务工厂Bean

总结

我们通过使用服务定位器模式实现了一种扩展 Spring 控制反转的绝妙方法。它帮助我们解决了依赖注入未提供最佳解决方案的用例。也就是说,依赖注入仍然是首选,并且在大多数情况下不应使用服务定位器来替代依赖注入。



欢迎加入我的知识星球,一起探讨架构,交流源码。加入方式,长按下方二维码噢

已在知识星球更新源码解析如下:

最近更新《芋道 SpringBoot 2.X 入门》系列,已经 101 余篇,覆盖了 MyBatis、Redis、MongoDB、ES、分库分表、读写分离、SpringMVC、Webflux、权限、WebSocket、Dubbo、RabbitMQ、RocketMQ、Kafka、性能测试等等内容。

提供近 3W 行代码的 SpringBoot 示例,以及超 4W 行代码的电商微服务项目。

获取方式:点“在看”,关注公众号并回复 666 领取,更多内容陆续奉上。

文章有帮助的话,在看,转发吧。

谢谢支持哟 (*^__^*)

微信扫码关注该文公众号作者

戳这里提交新闻线索和高质量文章给我们。
相关阅读
唱 三 歌游戏公司精确测算:在真实项目中采用AI画图节省了80%美术成本!0.37%概率奇迹!他拿下该项目中国首个世冠Spring Boot 实现跨域的 5 种方式,总有一种适合你,建议收藏因丈夫一句话连生7娃,还个个是学霸!?看了这位妈妈的养娃模式,服气了发改委《关于做好推进有效投资重要项目中废旧设备规范回收利用工作的通知》0.37%概率奇迹!他击败韩国选手,拿下这个项目中国首个世界冠军!很多人都看哭了…闲话人生(223)深切怀念我亲爱的二姐Offer捷报 l 约翰霍普金斯大学市场营销Offer+1!Rutgers学长成功申请综排Top10名校Marketing项目!真厉害!掌握这种思维的逻辑宝贝,让人刮目相看邓小平三起三落如何玩弄华国锋真正厉害的人,早就把人生调成了这三种模式天赋“易昺(bǐng)”,创造历史!希望国家再次严控管理对吗?深入剖析 Spring Boot 的 SPI 机制CTO:谁在项目中使用Arrays.asList、ArrayList.subList,就立马滚蛋!国企平台收购上市公司的五种模式二月的悉尼有多美?从这些游玩的项目中就能体会!悉尼,一座定居了就不想离开的城市!西雅图周末不无聊|华州春天最盛大的庆祝活动Washington Spring Fair来啦!2022年去世的15位老海归SpringBoot 整合 ChatGPT API 项目实战真事!澳男裸体在家,被进门偷窃的小偷撞上!两人四目相对后,小偷竟然干出这种事!我让ChatGPT复盘病例,它的回答让我刮目相看CTO:禁止在项目中使用 BeanUtils 属性转换工具。。。因为它,我从此对张家辉刮目相看颜纯鈎: 细颈瓶才是中国人痛苦的罪魁祸首别只会 SpringMVC 了!Spring 又官宣了一个更牛逼的替代框架!终于见识到了五彩斑斓的黑!中式高奢又一次让我刮目相看…9岁乔治小王子成熟了,待人接物令人刮目,头发浓密像戴妃!夏洛特也被查尔斯宠爱,将获高大上头衔!GreenNews播报|东京湾eSG项目计划建成世界首个ESG城市实用英语情景对话|English speaking practiceAgustín Hernández:中美洲建筑背景下的未来主义巨构管理对新手友好的开源社区的三个步骤 | Linux 中国一文详解Prompt学习和微调(Prompt Learning & Prompt Tuning)Hadoop/Spark 太重,esProc SPL 很轻
logo
联系我们隐私协议©2024 redian.news
Redian新闻
Redian.news刊载任何文章,不代表同意其说法或描述,仅为提供更多信息,也不构成任何建议。文章信息的合法性及真实性由其作者负责,与Redian.news及其运营公司无关。欢迎投稿,如发现稿件侵权,或作者不愿在本网发表文章,请版权拥有者通知本网处理。