背景
  
  同一个接口有多种实现,项目启动时按某种规则来选择性的启用其中一种实现,再具体一点,比如Controller初始化的时候,根据配置文件的指定的实现类前缀,来记载具体Service,不同Service使用不同的Dao和数据库。
  
  看到这里,我们会想到使用SPI机制,或Spring按条件加载Bean机制来实现,下面主要讨论后者。
  
  定义接口
  
  定义2个Service层接口:OrderService、OrderPromotionService,分别有一个方法,如下:
  
  // OrderService.java
  
  public interface OrderService {
  
  /**
  
  * 通过tid查询订单信息
  
  * @param tid 订单主键
  
  */
  
  List<Order> findByTid(Long tid);
  
  }
  
  ...
  
  // OrderPromotionService.java
  
  public interface OrderPromotionService {
  
  /**
  
  * 通过tid获取促销详情.
  
  * @param tid 订单唯一标识
  
  */
  
  默认实现
  
  分别实现上面2个接口的各自的方法,包路径:com.a.b。
  
  为达到根据规则装载不同ServiceImpl的目的,需要使用@Conditional注解,并且实现规则定义DefaultCondition。当Spring扫描到@Service注解时,会判断DefaultCondition#matches()方法,决定是否装载该ServiceImpl。
  
  DefaultCondition实现如下:
  
  // DefaultCondition.java
  
  public class DefaultCondition extends ParentCondition implements Condition {
  
  @Override
  
  public boolean matches(ConditionContext conditionContext, AnnotatedTypeMetadata annotatedTypeMetadata) {
  
  Environment environment = conditionContext.getEnvironment();
  
  String interfaceName = getInterFaceName(annotatedTypeMetadata);
  
  String implPrefix = environment.getProperty(interfaceName);
  
  if (StringUtils.isEmpty(implPrefix)) {
  
  return true;
  
  }
  
  return Constant.DEFAULT_PREFIX.equals(implPrefix);
  
  }
  
  }
  
  ...
  
  // Constant.java
  
  public class Constant {
  
  public static final String THIRD_PARTY_PREFIX = "ThirdParty";
  
  public static final String DEFAULT_PREFIX = "Default";
  
  DefaultCondition实现了Spring的Condition接口,先通过方法ParentCondition#getInterFaceName()获取ServiceImpl实现的接口名称interfaceName,然后从配置文件中获取该接口指定的实现类的前缀implPrefix,然后判断是否为Constant.DEFAULT_PREFIX,如果是,则装载该ServiceImpl。
  
  获取接口名称的实现定义在ParentCondition类中(是自己实现的),是这里需要继承ParentCondition的原因。
  
  Service层实现
  
  // DefaultOrderServiceImpl.java
  
  @Conditional(DefaultCondition.class)
  
  @Service
  
  public class DefaultOrderServiceImpl implements OrderService {
  
  @Autowired
  
  private OrderDao orderDao;
  
  @Override
  
  public List<Order> findByTid(Long tid) {
  
  return orderDao.findByTid(tid);
  
  }
  
  }
  
  ...
  
  /**
  
  * 默认实现.
  
  */
  
  @Conditional(DefaultCondition.class)
  
  @Service
  
  public class DefaultOrderPromotionServiceImpl implements OrderPromotionService {
  
  @Autowired
  
  private OrderPromotionDao promotionDao;
  
  @Override
  
  public List<OrderPromotion> findByTid(Long tid) {
  
  Dao层实现
  
  // OrderDao.java
  
  @Repository
  
  public class OrderDao {
  
  @Autowired
  
  @Qualifier("firstJdbcTemplate")
  
  private JdbcTemplate jdbcTemplate;
  
  public List<Order> findByTid(Long tid) {
  
  // 省略...
  
  }
  
  }
  
  ...
  
  @Repository
  
  public class OrderPromotionDao {
  
  @Autowired
  
  @Qualifier("firstJdbcTemplate")
  
  private JdbcTemplate jdbcTemplate;
  
  public List<OrderPromotion> findByTid(final Long tid) {
  
  // 省略...
  
  Dao层使用的是自己的数据源,代码注入的是firstJdbcTemplate,Spring-Boot多数据源配置不是今天讨论的重点,这里不再详细说明。
  
  第三方实现
  
  这里假设上面的2个Service接口还有一种第三方的实现,包路径:com.c.d。跟默认实现类似,这里也要定义自己的ServiceImpl装载规则ThirdPartyCondition,实现如下:
  
  // ThirdPartyOrderServiceImpl.java
  
  public class ThirdPartyCondition extends ParentCondition implements Condition {
  
  @Override
  
  public boolean matches(ConditionContext conditionContext, AnnotatedTypeMetadata annotatedTypeMetadata) {
  
  Environment environment = conditionContext.getEnvironment();
  
  String interfaceName = getInterFaceName(annotatedTypeMetadata);
  
  String implPrefix = environment.getProperty(interfaceName);
  
  return Constant.THIRD_PARTY_PREFIX.equals(implPrefix);
  
  先获取ServiceImpl实现的接口名称interfaceName,然后从配置文件中获取该接口指定的实现类的前缀implPrefix,然后判断是否为Constant.THIRD_PARTY_PREFIX,如果是,则装载该ServiceImpl。
  
  Service层实现
  
  // ThirdPartyOrderServiceImpl.java
  
  @Conditional({ThirdPartyCondition.class})
  
  @Service
  
  public class ThirdPartyOrderServiceImpl implements OrderService {
  
  @Autowired
  
  private ThirdPartyOrderDao thirdPartyOrderDao;
  
  @Override
  
  public List<Order> findByTid(Long tid) {
  
  return thirdPartyOrderDao.findByTid(tid);
  
  }
  
  }
  
  ...
  
  //ThirdPartyOrderPromotionServiceImpl.java
  
  @Conditional(ThirdPartyCondition.class)
  
  @Service
  
  public class ThirdPartyOrderPromotionServiceImpl implements OrderPromotionService {
  
  @Autowired
  
  private ThirdPartyOrderPromotionDao thirdPartyOrderPromotionDao;
  
  @Override
  
  public List<OrderPromotion> findByTid(Long tid) {
  
  Dao层实现
  
  省略,Dao层实现使用另一个数据源,注入secondJdbcTemplate。
  
  以上分别为OrderService和OrderPromotionService提供了两种实现,如下:
  
  Service接口 Service默认实现 Service第三方实现
  
  OrderService DefaultOrderServiceImp
  
  使用Dao层OrderDao,数据源first www.michenggw.com JdbcTemplate ThirdPartyOrderServiceImpl
  
  使用Dao层ThirdPartyOrderDao,数据源secondJdbcTemplate
  
  OrderPromotionService DefaultOrderPromotionServiceImpl
  
  使用Dao层DefaultOrderPromotionServiceImpl,数据源firstJdbcTemplate ThirdPartyOrderPromotionServiceImpl
  
  使用Dao层ThirdPartyOrderPromotionDao,数据源secondJdbcTemplate
  
  配置规则
  
  ServiceImpl定义完成,装载规则也定义了,下面我们在Spring-Boot中分别指定两个类的加载对象
  
  // application.properties
  
  com.free.spring.jdbc.demo.service.OrderPromotionService=ThirdParty
  
  com.free.spring.jdbc.demo.service.OrderService=Default
  
  如上,定义了OrderPromotionService接口使用第三方的实现,OrderService接口使用默认实现
  
  下面简单的写两个Controller看一下效果。
  
  运行效果
  
  1. 数据准备
  
  为first库的2张表分别添加1条数据,如下:
  
  为second库的2张表添加数据
  
  2. Controller定义
  
  // OrderController.java
  
  @RestController
  
  @RequestMapping(www.dasheng178.com"/order")
  
  public class OrderController {
  
  @Autowired
  
  private OrderService orderService;
  
  @GetMapping("/{tid}")
  
  @ResponseBody
  
  public List<Order>www.haom178.com findByTid(@PathVariable(www.gouyiflb.cn"tid") Long tid) {
  
  return orderService.findByTid(tid);
  
  }
  
  }
  
  ...
  
  // OrderPromotionController.java
  
  @RestController
  
  @RequestMapping("/promotion")
  
  public class OrderPromotionController {
  
  @Autowired
  
  private OrderPromotionService promotionService;
  
  @GetMapping("www.gcyl159.com/ /{tid}")
  
  @ResponseBody
  
  public List<OrderPromotion> query(@PathVariable("tid") Long tid) {
  
  try {
  
  return promotionService.findByTid(tid);
  
  } catch (Exception e) {
  
  e.printStackTrace();
  
  }
  
  return null;
  
  我们通过Controller分别请求OrderService和OrderPromotionService,通过返回的数据判断是否真的实现了选择性装载ServiceImpl。
  
  然后我们分别请求上面两个接口,观察结果:
  
  OK!没问题!
  
  如果大家有更简单的方式,欢迎探讨~

Spring-Boot基于配置按条件装Bean的更多相关文章

  1. Spring boot将配置属性注入到bean类中

    一.@ConfigurationProperties注解的使用 看配置文件,我的是yaml格式的配置: // file application.yml my: servers: - dev.bar.c ...

  2. Spring boot将配置属性注入到bean 专题

    https://blog.csdn.net/wangmx1993328/article/details/81002901 Error starting ApplicationContext. To d ...

  3. 分布式事务、多数据源、分库分表中间件之spring boot基于Atomikos+XADataSource分布式事务配置(100%纯动态)

    本文描述spring boot基于Atomikos+DruidXADataSource分布式事务配置(100%纯动态),也就是增加.减少数据源只需要修改application.properties文件 ...

  4. Spring Boot自动配置与Spring 条件化配置

    SpringBoot自动配置 SpringBoot的自动配置是一个运行时(应用程序启动时)的过程,简化开发时间,无需浪费时间讨论具体的Spring配置,只需考虑如何利用SpringBoot的自动配置即 ...

  5. Spring boot 基于注解方式配置datasource

    Spring boot 基于注解方式配置datasource 编辑 ​ Xml配置 我们先来回顾下,使用xml配置数据源. 步骤: 先加载数据库相关配置文件; 配置数据源; 配置sqlSessionF ...

  6. Sping Boot入门到实战之入门篇(四):Spring Boot自动化配置

    该篇为Sping Boot入门到实战系列入门篇的第四篇.介绍Spring Boot自动化配置的基本原理与实现.   Spring Boot之所以受开发者欢迎, 其中最重要的一个因素就是其自动化配置特性 ...

  7. Springboot 系列(三)Spring Boot 自动配置原理

    注意:本 Spring Boot 系列文章基于 Spring Boot 版本 v2.1.1.RELEASE 进行学习分析,版本不同可能会有细微差别. 前言 关于配置文件可以配置的内容,在 Spring ...

  8. Spring Boot自动配置原理与实践(一)

    前言 Spring Boot众所周知是为了简化Spring的配置,省去XML的复杂化配置(虽然Spring官方推荐也使用Java配置)采用Java+Annotation方式配置.如下几个问题是我刚开始 ...

  9. Spring Boot自动配置原理(转)

    第3章 Spring Boot自动配置原理 3.1 SpringBoot的核心组件模块 首先,我们来简单统计一下SpringBoot核心工程的源码java文件数量: 我们cd到spring-boot- ...

随机推荐

  1. dubbo之main启动

    一.dubbo的main启动在使用上面会简单的多,但是需要做一些简单的配置. dubbo.spring.config=classpath*:META-INF/spring/*.xml 备注:这个是默认 ...

  2. L011系统文件属性知识进阶详解小节

    L011系统文件属性知识进阶详解小节 这节课的内容相对来说较少,一上午加中午就听完了,现在总结一下,最后会有一个相关的面试题. 首先先附上一张图: 今天学习主要跟①和②有关,①为Inode 号 ②为文 ...

  3. shell 批量压缩指定目录及子目录内图片的方法

    用户上传的图片,一般都没有经过压缩,造成空间浪费.因此需要编写一个程序,查找目录及子目录的图片文件(jpg,gif,png),将大于某值的图片进行压缩处理. 查看目录文件大小 du -h --max- ...

  4. JQuery.extend扩展实现同步post请求

    有时需要在jQuery中实现同步post请求,而jquery自带的是异步,需要通过JQuery.extend扩展. 支持ie和firefox,方法转载而来.需要在submit前将form.append ...

  5. Linux命令应用大词典-第34章 打印与传真

    34.1 lpr:打印文件 34.2 lpq:显示打印队列状态 34.3 lprm:取消打印作业 34.4 lpstat:显示cups状态信息 34.5 cupsaccept:接受作业发送到目的地 3 ...

  6. 【转】跨平台移动端开发框架NativeScript 发布正式版本

    原文:http://news.cnblogs.com/n/520865/ Nativescript 项目地址:http://www.telerik.com/nativescript “一次编码,处处运 ...

  7. ionic ios样式偏移解决方案。

    在css属性内增加: .item-ios [item-end] { //解决ios系统上尾部图标出现重影而增加的格式. margin: 0px -15.3px 0px 0px; margin-bott ...

  8. Python基础 之 tuple类-元组 和 dict类-字典

    tuple 元组 一.tuple 类的基本属性 1.元组,有序:元素不可被修改,不能被增加或者删除tuple类 tu = (111,22,33,44) 一般写元组的时候,推荐在最后加入,和类方法进行区 ...

  9. Hadoop学习(一):完全分布式集群环境搭建

    1. 设置免密登录 (1) 新建普通用户hadoop:useradd hadoop(2) 在主节点master上生成密钥对,执行命令ssh-keygen -t rsa便会在home文件夹下生成 .ss ...

  10. BZOJ 4815 CQOI2017 小Q的表格 欧拉函数+分块

    题目链接:https://www.lydsy.com/JudgeOnline/problem.php?id=4815 题意概述:要认真概述的话这个题就出来了... 分析: 首先分析题目,认真研究一下修 ...