Spring 环境下实现策略模式的示例

 更新时间:2020年10月12日 09:49:57   作者:CodingDiary  
这篇文章主要介绍了Spring 环境下实现策略模式的示例,帮助大家更好的理解和使用spring框架,感兴趣的朋友可以了解下

背景

最近在忙一个需求,大致就是给满足特定条件的用户发营销邮件,但是用户的来源有很多方式:从 ES 查询的、从 csv 导入的、从 MongoDB 查询….. 需求很简单,但是怎么写的优雅,方便后续扩展,就存在很多门道了。

我们的项目是基于 Spring Boot 开发的,因此这篇文章也会基于 Spring Boot 作为基础框架,教你如何使用 Spring 依赖注入的特性,优雅的实现策略模式。

1. 简单粗暴

最简单粗暴直接的方式莫过于 if...else… 了,伪代码如下:

if(来源 == ES){
 // TODO: ES Query
}else if(来源 == CSV){
 // TODO: Read CSV File
}else if(来源 == MongoDB){
 // TODO: MongoDB Query
}

如果后面还需要从其他平台获取,那就在接着添加 else if...,这种方式固然简单直接,但是当后续扩展的方式越来越多,相应的if...else...也会越来越长,emmm….. 怎么说呢,黑猫白猫,能抓到老鼠的就是好猫。

2. 策略模式

在 Spring 环境下实现策略模式异常简单,毕竟 Spring 提供的依赖注入简直就是开发利器~

既然是策略模式,那么定义策略肯定是首当其冲,策略我们使用枚举实现最佳。

public enum GroupType {
  /**
   * 从 ES 查询
   */
  ES,
  /**
   * 从 MongoDB 查询
   */
  MONGODB,
  /**
   * 从 文件 读取
   */
  FILE
}

下一步,我们定义一个接口,用于抽象通用的功能。

public interface IGroupSelect {
  /**
   * 人群获取方式
   *
   * @return 人群选择枚举
   */
  GroupType type();

  /**
   * 查询满足条件的用户
   *
   * @param groupQuery 查询条件
   * @return 满足条件的用户
   */
  default List<GroupUser> queryUser(GroupQuery groupQuery) {
    checkQueryCondition(groupQuery);
    return doQuery(groupQuery);
  }

  /**
   * 事前校验查询条件
   *
   * @param groupQuery 查询条件
   * @throws IllegalArgumentException 参数异常
   */
  void checkQueryCondition(GroupQuery groupQuery) throws IllegalArgumentException;

  /**
   * 真正的查询方法
   *
   * @param groupQuery 查询条件
   * @return 满足条件的用户
   */
  List<GroupUser> doQuery(GroupQuery groupQuery);

}

这一步,小伙伴们有没有发现里面也包含了模板方法模式呢?

然后就是不同策略的具体实现了。

  • ES 策略
@Slf4j
@Service
public class EsGroupSelect implements IGroupSelect {

  /**
   * 人群获取方式
   *
   * @return 人群选择枚举
   */
  @Override
  public GroupType type() {
    return GroupType.ES;
  }

  /**
   * 事前校验查询条件
   *
   * @param groupQuery 查询条件
   * @throws IllegalArgumentException 参数异常
   */
  @Override
  public void checkQueryCondition(GroupQuery groupQuery) throws IllegalArgumentException {
    log.info("groupQuery = {}", groupQuery);
  }

  /**
   * 查询满足条件的用户
   *
   * @param groupQuery 查询条件
   * @return 满足条件的用户
   */
  @Override
  public List<GroupUser> doQuery(GroupQuery groupQuery) {
    List<GroupUser> result = new ArrayList<>();
    // TODO:
    // 1. 复杂的 ES 查询逻辑
    // 2. 根据条件筛选满足条件的用户数据
    for (int i = 1; i <= 15; i++) {
      result.add(GroupUser.of("ES用户" + i, i + "@es.com"));
    }
    return result;
  }
}
  • 文件策略
@Slf4j
@Service
public class FileGroupSelect implements IGroupSelect {
  /**
   * 人群获取方式
   *
   * @return 人群选择枚举
   */
  @Override
  public GroupType type() {
    return GroupType.FILE;
  }

  /**
   * 事前校验查询条件
   *
   * @param groupQuery 查询条件
   * @throws IllegalArgumentException 参数异常
   */
  @Override
  public void checkQueryCondition(GroupQuery groupQuery) throws IllegalArgumentException {
    log.info("groupQuery = {}", groupQuery);
  }

  /**
   * 查询满足条件的用户
   *
   * @param groupQuery 查询条件
   * @return 满足条件的用户
   */
  @Override
  public List<GroupUser> doQuery(GroupQuery groupQuery) {
    List<GroupUser> result = new ArrayList<>();
    // TODO:
    // 1. 复杂的解析、读文件
    // 2. 根据条件筛选满足条件的用户数据
    for (int i = 1; i <= 3; i++) {
      result.add(GroupUser.of("文件读取用户" + i, i + "@file.com"));
    }
    return result;
  }
}
  • MongoDB 策略
@Slf4j
@Service
public class MongoGroupSelect implements IGroupSelect {
  /**
   * 人群获取方式
   *
   * @return 人群选择枚举
   */
  @Override
  public GroupType type() {
    return GroupType.MONGODB;
  }

  /**
   * 事前校验查询条件
   *
   * @param groupQuery 查询条件
   * @throws IllegalArgumentException 参数异常
   */
  @Override
  public void checkQueryCondition(GroupQuery groupQuery) throws IllegalArgumentException {
    log.info("groupQuery = {}", groupQuery);
  }

  /**
   * 查询满足条件的用户
   *
   * @param groupQuery 查询条件
   * @return 满足条件的用户
   */
  @Override
  public List<GroupUser> doQuery(GroupQuery groupQuery) {
    List<GroupUser> result = new ArrayList<>();
    // TODO:
    // 1. 复杂的 MongoDB 查询逻辑
    // 2. 根据条件筛选满足条件的用户数据
    for (int i = 1; i <= 7; i++) {
      result.add(GroupUser.of("MongoDB用户" + i, i + "@mongo.com"));
    }
    return result;
  }
}

现在到了最后一步,就是如何通过 Spring 优雅的实现策略模式的选择呢?敲黑板,考试必考!

我们通过定义一个工厂类,然后使用 Spring 的依赖注入特性,可以注入一个接口的多个实现,这里采用 List<IGroupSelect> 的形式注入,Spring 也支持通过 Map<String,IGroupSelect> 的形式注入,如果使用 Map 注入,那么 key 就是类名,小伙伴们自己也可以测试一下~

@Service
public class GroupSelectFactory {
  @Autowired
  private List<IGroupSelect> groupSelectList;

  /**
   * 根据人群类型选择具体的实现类
   *
   * @param type 人群类型
   * @return 人群选择具体实现类
   */
  public IGroupSelect getGroupSelect(GroupType type) {
    Optional<IGroupSelect> groupSelectOptional = groupSelectList.stream().filter(t -> t.type() == type).findAny();
    return groupSelectOptional.orElseThrow(() -> new IllegalArgumentException("暂不支持该人群方式"));
  }
}

最后写个定时任务测试一下吧。

@Autowired
private GroupSelectFactory groupSelectFactory;

/**
 * 模拟定时发送营销邮件
 */
@Scheduled(cron = "0/10 * * * * ?")
public void sendEmailTask() {
  List<SendEmailTask> taskList = new ArrayList<>();
  for (GroupType groupType : GroupType.values()) {
    GroupQuery groupQuery = new GroupQuery("虚头巴脑的 " + groupType.name() + " 查询条件");
    taskList.add(SendEmailTask.of(groupType, groupQuery));
  }

  taskList.forEach(task -> {
    List<GroupUser> groupUsers = groupSelectFactory.getGroupSelect(task.getType()).queryUser(task.getQuery());
    log.info("groupUsers = {}", groupUsers);
  });

}

@Data
@NoArgsConstructor
@AllArgsConstructor(staticName = "of")
static class SendEmailTask implements Serializable {
  private static final long serialVersionUID = -3461263089669779193L;
  private GroupType type;
  private GroupQuery query;
}

观察控制台,看看日志输出吧~

总结

  • 本文使用策略模式实现不同人群的查询,后续如果要增加短信、微信、钉钉的消息发送,是不是也可以用策略模式实现呢?
  • 使用 Spring 的依赖注入特性,可以注入一个接口的多个实现,很容易就实现了策略模式的选择,这样后续添加一种策略的时候,完全不需要改动主要逻辑,只需添加具体实现即可。
  • 细心的小伙伴可以发现,本文虽然是讲策略模式,其实里面还包含了模板方法、工厂模式,多种设计模式的协同作战,食用味道更佳哟~

配套代码:https://github.com/xkcoding/practice_demo/tree/master/strategy-design-pattern-in-spring

以上就是Spring 环境下实现策略模式的示例的详细内容,更多关于Spring 实现策略模式的资料请关注脚本之家其它相关文章!

相关文章

  • 使用Mybatis实现分页效果示例

    使用Mybatis实现分页效果示例

    大家好,本篇文章主要讲的是使用Mybatis实现分页效果示例,感兴趣的同学赶快来看一看吧,对你有帮助的话记得收藏一下,方便下次浏览
    2021-12-12
  • mybatis通过中间表实现一对多查询功能

    mybatis通过中间表实现一对多查询功能

    这篇文章主要介绍了mybatis通过中间表实现一对多查询,通过一个学生的id查询出该学生所学的所有科目,本文通过实例代码给大家介绍的非常详细,需要的朋友可以参考下
    2021-08-08
  • 浅谈java实现mongoDB的多条件查询

    浅谈java实现mongoDB的多条件查询

    这篇文章主要介绍了java实现mongoDB的多条件查询,具有一定参考价值,需要的朋友可以参考下。
    2017-09-09
  • 详解Java中的do...while循环语句的使用方法

    详解Java中的do...while循环语句的使用方法

    这篇文章主要介绍了Java中的do...while循环语句的使用方法,是Java入门学习中的基础知识,需要的朋友可以参考下
    2015-10-10
  • Gauva使用ListenableFuture介绍说明

    Gauva使用ListenableFuture介绍说明

    并发是一个困难问题,但是通过强大和强大的抽象能够显著的简化工作。为了简化问题,Gauva使用ListenableFuture扩展了JDK的Future接口,这篇文章主要介绍了Gauva使用ListenableFuture
    2023-01-01
  • SpringBoot使用@PathVariable进行数据校验的流程步骤

    SpringBoot使用@PathVariable进行数据校验的流程步骤

    在SpringBoot项目中,我们经常需要从 URL 中获取参数并进行相关的数据校验,而@PathVariable注解就是一种非常方便的方式,可以让我们在方法参数中直接获取URL中的参数,并进行数据校验,本文将介绍如何使用@PathVariable注解进行数据校验
    2023-06-06
  • java8 stream sort自定义复杂排序案例

    java8 stream sort自定义复杂排序案例

    这篇文章主要介绍了java8 stream sort自定义复杂排序案例,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2020-10-10
  • 详解Mybatis的二级缓存配置

    详解Mybatis的二级缓存配置

    这篇文章主要介绍了Mybatis的二级缓存配置的相关资料,需要的朋友可以参考下
    2017-05-05
  • Spring:@Async注解和AsyncResult与CompletableFuture使用问题

    Spring:@Async注解和AsyncResult与CompletableFuture使用问题

    这篇文章主要介绍了Spring:@Async注解和AsyncResult与CompletableFuture使用问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2024-08-08
  • 如何使用JJWT及JWT讲解和工具类

    如何使用JJWT及JWT讲解和工具类

    关于JWT的文章网上已经多如牛毛了,但是相信很多同学学的还是云里雾里,所以在我学习JWT之后尽量用最简洁的描述写下这篇文章用于日后复习,与此同时也希望可以帮助同学们共同进步
    2021-09-09

最新评论