0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

Spring项目中用这种模式更方便

jf_78858299 来源:JAVA旭阳 作者:JAVA旭阳 2023-05-11 10:39 次阅读

前言

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

文件解析器的例子

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

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

  1. 定义一个类型的枚举
public enum ContentType {
  JSON,
  CSV
}
  1. 定义一个解析的接口
public interface Parser {
  List parse(Reader r);
}
  1. 根据不同的文件类型有不同的实现类
// 解析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根据不同的类型调用不同的实现
@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中添加新的类型,这就导致客户端和不同的解析器紧密耦合。

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

应用Service Locator Pattern

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

  1. 让我们定义我们的服务定位器接口ParserFactory, 它有一个接受内容类型参数并返回Parser的方法。
public interface ParserFactory {
  Parser getParser(ContentType contentType);
}
  1. 我们配置ServiceLocatorFactoryBean使用ParserFactory作为服务定位器接口,ParserFactory这个接口不需要写实现类。
@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);
  }
  ..
}

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

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

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

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

图片

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

总结

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

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • 接口
    +关注

    关注

    33

    文章

    8444

    浏览量

    150709
  • JSON
    +关注

    关注

    0

    文章

    116

    浏览量

    6929
  • csv
    csv
    +关注

    关注

    0

    文章

    38

    浏览量

    5797
收藏 人收藏

    评论

    相关推荐

    Gradle构建的多模块Spring Boot项目

    构建的多模块 Spring Boot 项目,JDK 版本为8,IDEA 版本为 2022.3.1 。 1. 创建项目 打开IDEA,选择菜单:File -> New -> Project,在其
    的头像 发表于 09-25 14:46 2790次阅读
    Gradle构建的多模块<b class='flag-5'>Spring</b> Boot<b class='flag-5'>项目</b>

    java spring教程

    java spring教程理解Spring 实现原理掌握Spring IOC,AOP掌握Spring的基础配置和用法熟练使用SSH开发项目
    发表于 09-11 11:09

    什么是java spring

    。在SSH项目中管理事务以及对象的注入Spring是非侵入式的:基于Spring开发的系统中的对象一般不依赖于Spring的类。组成 Spring
    发表于 09-11 11:16

    如何把自己做的项目中用到的所有子vi打包成库类?

    如题,如何将自己做的项目中的所有子vi打包成库类,方便以后用到的时候调用?因为现在做了挺多的项目,其中有很多都是用的写好的功能的子vi,每次写新的项目的时候都要重新复制到新的
    发表于 11-23 16:33

    Spring MVC练手项目

    初识 Spring MVC——练手小项目
    发表于 09-17 08:41

    如何在我的项目中使用停止模式

    你好,我想在我的项目中使用停止模式。有什么例子吗?我想让我的外围模块在初始化时停止模式。如果用户将唤醒按钮,模块醒来并开始广告。模块进入停止模式,再然后preconfiguredtim
    发表于 09-25 14:58

    启动Spring Boot项目应用的三种方法

    方便。打个比方,如果我们做传统的spring web项目,我们需要做哪些工作。1)配置web.xml,加载springspring mv
    发表于 01-14 17:33

    Spring认证」Spring Hello World 项目示例

    。现在使用向导窗口将您的项目命名为HelloSpring,如下所示 -成功创建项目后,您的项目资源管理器中将包含以下内容-第 2 步 - 添加所需的库第二步,让我们在项目中添加
    发表于 08-17 13:49

    Spring认证_什么是Spring GraphQL

    Spring GraphQL 为构建在 GraphQL Java 上的 Spring 应用程序提供支持。两个团队之间的联合联合。我们的共同理念是少固执己见,专注于全面和广泛的支持。 Spri
    的头像 发表于 08-06 14:30 677次阅读
    <b class='flag-5'>Spring</b>认证_什么是<b class='flag-5'>Spring</b> GraphQL

    Spring认证」什么是Spring GraphQL?

    这个项目建立在 Boot 2.x 上,但它应该与最新的 Boot2.4.x5 相关。 要创建项目,请转到start.spring.io并为要使用的GraphQL传输选择启动器: 启动机 运输 执行
    的头像 发表于 08-10 14:08 785次阅读
    「<b class='flag-5'>Spring</b>认证」什么是<b class='flag-5'>Spring</b> GraphQL?

    机器视觉项目中需用到的光源

    不同效果 。 根据我们的统计,机器视觉项目中用到的光源有如下几种: 环形光源(LQ-HDRmmnn-C): 光出射角度值在0°~90°。0°~45°为低角度环形光源,目前应用案例包括手机金属外框划痕检测、光滑表面的划痕、破损检测以达到突显物体轮廓
    的头像 发表于 05-30 09:26 869次阅读

    Spring中用到了哪些设计模式

    Spring 通过动态代理对类进行方法级别的切面增强,动态生成目标对象的代理类,并在代理类的方法中设置拦截器,通过执行拦截器中的逻辑增强了代理方法的功能,从而实现 AOP。
    发表于 05-30 09:43 286次阅读
    <b class='flag-5'>Spring</b><b class='flag-5'>中用</b>到了哪些设计<b class='flag-5'>模式</b>

    kafka client在 spring如何实现

    之前写过关于 Apache Pulsar 的简单示例,用来了解如何使用 Pulsar 这个新生代的消息队列中间件,但是如果想要在项目中使用,还会欠缺很多,最明显的就是 集成复杂,如果你用过其他
    的头像 发表于 09-25 11:21 451次阅读
    kafka client在 <b class='flag-5'>spring</b>如何实现

    Spring Boot的启动原理

    spring-boot-maven-plugin 的 maven 项目打包插件,可以方便的将 Spring Boot 项目打成 jar 包
    的头像 发表于 10-13 11:44 596次阅读
    <b class='flag-5'>Spring</b> Boot的启动原理

    Spring中经典的9种设计模式

    spring中常用的设计模式达到九种,我们一一举例
    的头像 发表于 12-11 09:56 896次阅读
    <b class='flag-5'>Spring</b>中经典的9种设计<b class='flag-5'>模式</b>