来源:JAVA旭阳
不知道大家在项目中有没有遇到过这样的场景,根据传入的类型,调用接口不同的实现类或者说服务,比如根据文件的类型使用 CSV解析器或者JSON解析器,在调用的客户端一般都是用if else去做判断,比如类型等于JSON,我就用JSON解析器,那如果新加一个类型的解析器,是不是调用的客户端还要修改呢?这显然太耦合了,本文就介绍一种方法,服务定位模式Service Locator Pattern来解决,它帮助我们消除紧耦合实现及其依赖性,并提出将服务与其具体类解耦。
文件解析器的例子
我们通过一个例子来告诉你如何使用Service Locator Pattern。
假设我们有一个从各种来源获取数据的应用程序,我们必须解析不同类型的文件,比如解析CSV文件和JSON文件。
1、定义一个类型的枚举
publicenumContentType{ JSON, CSV }
2、定义一个解析的接口
publicinterfaceParser{ Listparse(Readerr); }
3、根据不同的文件类型有不同的实现类
//解析csv @Component publicclassCSVParserimplementsParser{ @Override publicListparse(Readerr){..} } //解析json @Component publicclassJSONParserimplementsParser{ @Override publicListparse(Readerr){..} }
4、最后写一个调用的客户端,通过switch case根据不同的类型调用不同的实现
@Service publicclassClient{ privateParsercsvParser,jsonParser; @Autowired publicClient(ParsercsvParser,ParserjsonParser){ this.csvParser=csvParser; this.jsonParser=jsonParser; } publicListgetAll(ContentTypecontentType){ .. switch(contentType){ caseCSV: returncsvParser.parse(reader); caseJSON: returnjsonParser.parse(reader); .. } } .. }
可能大部分人都是像上面一样的方式实现的,也能正常运行,那深入思考下,存在什么问题吗?
现在假如产品经理提出了一个新需求要支持XML类型的文件,是不是客户端也要修改代码,需要在switch case中添加新的类型,这就导致客户端和不同的解析器紧密耦合。
那么有什么更好的方法呢?
应用Service Locator Pattern
没错,那就是用上我们的服务定位模式Service Locator Pattern。
1、让我们定义我们的服务定位器接口ParserFactory, 它有一个接受内容类型参数并返回Parser的方法。
publicinterfaceParserFactory{ ParsergetParser(ContentTypecontentType); }
2、我们配置ServiceLocatorFactoryBean使用ParserFactory作为服务定位器接口,ParserFactory这个接口不需要写实现类。
@Configuration publicclassParserConfig{ @Bean("parserFactory") publicFactoryBeanserviceLocatorFactoryBean(){ ServiceLocatorFactoryBeanfactoryBean=newServiceLocatorFactoryBean(); //设置服务定位接口 factoryBean.setServiceLocatorInterface(ParserFactory.class); returnfactoryBean; } }
3、设置解析器Bean的名称为类型名称,方便服务定位
//设置bean的名称和类型一致 @Component("CSV") publicclassCSVParserimplementsParser{..} @Component("JSON") publicclassJSONParserimplementsParser{..} @Component("XML") publicclassXMLParserimplementsParser{..}
4、修改枚举, 添加XML
publicenumContentType{ JSON, CSV, XML }
5、最后用客户端调用,直接根据类型调用对应的解析器,没有了switch case
@Service publicclassClient{ privateParserFactoryparserFactory; @Autowired publicClient(ParserFactoryparserFactory){ this.parserFactory=parserFactory; } publicListgetAll(ContentTypecontentType){ .. //关键点,直接根据类型获取 returnparserFactory .getParser(contentType) .parse(reader); } .. }
嘿嘿,我们已经成功地实现了我们的目标。现在再加新的类型,我们只要扩展添加新的解析器就行,再也不用修改客户端了,满足开闭原则。
如果你觉得Bean的名称直接使用类型怪怪的,这边可以建议你按照下面的方式来。
publicenumContentType{ JSON(TypeConstants.JSON_PARSER), CSV(TypeConstants.CSV_PARSER), XML(TypeConstants.XML_PARSER); privatefinalStringparserName; ContentType(StringparserName){ this.parserName=parserName; } @Override publicStringtoString(){ returnthis.parserName; } publicinterfaceTypeConstants{ StringCSV_PARSER="csvParser"; StringJSON_PARSER="jsonParser"; StringXML_PARSER="xmlParser"; } } @Component(TypeConstants.CSV_PARSER) publicclassCSVParserimplementsParser{..} @Component(TypeConstants.JSON_PARSER) publicclassJSONParserimplementsParser{..} @Component(TypeConstants.XML_PARSER) publicclassXMLParserimplementsParser{..}
剖析Service Locator Pattern
通过前面的例子,想必大家基本知道服务定位器模式如何使用了吧,现在我们深入剖析下。
服务定位器模式消除了客户端对具体实现的依赖。以下引自Martin Fowler的文章总结了核心思想:“服务定位器背后的基本思想是拥有一个知道如何获取应用程序可能需要的所有服务的对象。因此,此应用程序的服务定位器将有一个在需要时返回“服务”的方法。”
Spring的ServiceLocatorFactoryBean实现了FactoryBean接口,创建了Service Factory服务工厂Bean。
总结
我们通过使用服务定位器模式实现了一种扩展 Spring 控制反转的绝妙方法。它帮助我们解决了依赖注入未提供最佳解决方案的用例。也就是说,依赖注入仍然是首选,并且在大多数情况下不应使用服务定位器来替代依赖注入。
审核编辑:汤梓红
-
接口
+关注
关注
33文章
8652浏览量
151458 -
文件
+关注
关注
1文章
569浏览量
24774 -
spring
+关注
关注
0文章
340浏览量
14361 -
JSON
+关注
关注
0文章
119浏览量
6983
原文标题:Spring项目中用了这种模式,经理对我刮目相看
文章出处:【微信号:芋道源码,微信公众号:芋道源码】欢迎添加关注!文章转载请注明出处。
发布评论请先 登录
相关推荐
评论