基于【策略模式】设计多渠道发送消息

前言:设计模式源于生活

策略模式的基本概念

策略模式将可变的部分从程序中抽象分离成算法接口,在该接口下分别封装一系列算法实现,并使他们可以相互替换,从而导致客户端程序独立于算法的改变。

策略模式应用场景

1.解决我多重if条件判断
2.有共同行为,但是有不同的业务逻辑(例如:支付模式[支持多种支付模式],直播线路模式[支持多种线路切换],消息发送渠道模式[支持多种消息渠道发送])

策略模式优缺点

优点:

1.扩展性良好
2.算法自由切换
3.更好的代码复用性

缺点:

1.策略类会随着扩展增多,不方便维护
2.需要了解更多的业务细节

策略模式流程图

1.工厂维护行为(FactoryContext)角色:定义出一个工厂维护对象的引用的上下文方法,也是对外的唯一入口
2.抽象策略行为(Strategy)角色:定义出一个具有共同行为的接口,这个角色通常由一个Java抽象类或者Java接口实现
3.具体策略执行(ConcreteStrategy)角色:具体执行者接到转发请求后,直接执行业务逻辑

接下来我将通过三种方式实现策略模式

通用maven依赖

<parent>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-parent</artifactId>
    <version>2.2.1.RELEASE</version>
</parent>

<dependencies>
    <dependency>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-web</artifactId>
    </dependency>

    <dependency>
        <groupId>org.projectlombok</groupId>
        <artifactId>lombok</artifactId>
        <version>1.18.12</version>
    </dependency>

    <dependency>
        <groupId>org.apache.commons</groupId>
        <artifactId>commons-lang3</artifactId>
        <version>3.11</version>
    </dependency>

    <dependency>
        <groupId>com.baomidou</groupId>
        <artifactId>mybatis-plus-boot-starter</artifactId>
        <version>3.3.2</version>
    </dependency>
    <!-- mysql 依赖 -->
    <dependency>
        <groupId>mysql</groupId>
        <artifactId>mysql-connector-java</artifactId>
    </dependency>

</dependencies>

通用工具类

@Component
public class SpringUtils implements ApplicationContextAware {

    private static ApplicationContext applicationContext;

    @Override
    public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
        this.applicationContext = applicationContext;
    }

    //获取applicationContext
    public static ApplicationContext getApplicationContext() {
        return applicationContext;
    }

    //通过name获取 Bean.
    public static Object getBean(String name){
        return getApplicationContext().getBean(name);
    }

    //通过class获取Bean.
    public static <T> T getBean(Class<T> clazz){
        return getApplicationContext().getBean(clazz);
    }

    //通过name,以及Clazz返回指定的Bean
    public static <T> T getBean(String name,Class<T> clazz){
        return getApplicationContext().getBean(name, clazz);
    }

}

基于静态内存模式,实现策略模式

定义抽象行为策略

public abstract class BehaviorStrategy {

    /**
     * 执行具体消息策略模式
     */
    protected abstract void specificMsgStrategy ();
}

kafka消息渠道

@Slf4j
public class KafkaStrategy extends BehaviorStrategy {

    @Override
    protected void specificMsgStrategy() {
        log.info("执行kafkaMQ消息模式发送消息");
    }
}

rabbit消息渠道

@Slf4j
public class RabbitStrategy extends BehaviorStrategy {
    @Override
    protected void specificMsgStrategy() {
        log.info("执行rabbitMQ消息模式发送消息");
    }
}

rocket消息渠道

@Slf4j
public class RocketStrategy extends BehaviorStrategy {

    @Override
    protected void specificMsgStrategy() {
        log.info("执行rocketMQ消息模式发送消息");
    }
}

工厂模式类,用于维护对象的引用

public class FactoryStrategy {

    /**
     * 存储策略
     */
    private static Map<String, BehaviorStrategy> stringBehaviorStrategyMap = new ConcurrentHashMap<>();

    /**
     * 初始化策略
     */
    static {
        stringBehaviorStrategyMap.put("kafka", new KafkaStrategy());
        stringBehaviorStrategyMap.put("rabbit", new RabbitStrategy());
        stringBehaviorStrategyMap.put("rocket", new RocketStrategy());
    }

    /**
     * 执行具体策略
     * @param strategyId
     * @return
     */
    public static BehaviorStrategy getStrategy(String strategyId) {
        //待优化,如果查询空,友好提示
        return stringBehaviorStrategyMap.get(strategyId);
    }
}

测试类

public class Test {
    public static void main(String[] args) {
        BehaviorStrategy kafka = FactoryStrategy.getStrategy("kafka");
        kafka.specificMsgStrategy();
    }
}

效果图,选择策略是kafka模式,那么执行的业务逻辑就是kafka啦

基于工厂+spring容器实现策略模式

定义抽象行为策略

public abstract class BehaviorStrategy {

    /**
     * 执行具体消息策略模式
     */
    public  abstract void specificMsgStrategy ();
}

kafka消息渠道

@Slf4j
@Component
public class KafkaStrategy extends BehaviorStrategy {

    @Override
    public  void specificMsgStrategy() {
        log.info("执行kafkaMQ消息模式发送消息");
    }
}

rabbit消息渠道

@Slf4j
@Component
public class RabbitStrategy extends BehaviorStrategy {
    @Override
    public  void specificMsgStrategy() {
        log.info("执行rabbitMQ消息模式发送消息");
    }
}

rocket消息渠道

@Slf4j
@Component
public class RocketStrategy extends BehaviorStrategy {

    @Override
    public  void specificMsgStrategy() {
        log.info("执行rocketMQ消息模式发送消息");
    }
}

工厂模式,通过将策略具体执行者,注入到容器中,工厂通过beanid到容器中执行不同的策略具体执行者,相比第一种就简单很多

@Component
public class FactoryStrategy {


    public <T> T getStrategy(String strategyId,Class<T> t) {
        if (StringUtils.isBlank(strategyId)) {
            return null;
        }
        T bean = SpringUtils.getBean(strategyId, t);
        return bean;
    }

}

controller类

@RestController
public class StrategyController {

    @Autowired
    private FactoryStrategy factoryStrategy;

    @GetMapping("/getStrategy")
    public void getStrategy(@RequestParam("strategyId") String strategyId) {
        BehaviorStrategy strategy = factoryStrategy.getStrategy(strategyId, BehaviorStrategy.class);
        strategy.specificMsgStrategy();
    }
}

效果图

访问:http://localhost:8080/getStrategy?strategyId=rabbitStrategy
当参数策略值为:rabbit具体执行者,那么工厂会自动到容器找到相应的执行者执行业务逻辑

基于spring+db+工厂实现策略模式

sql语句

SET NAMES utf8mb4;
SET FOREIGN_KEY_CHECKS = 0;

-- ----------------------------
-- Table structure for strategy
-- ----------------------------
DROP TABLE IF EXISTS `strategy`;
CREATE TABLE `strategy`  (
  `id` int(11) NOT NULL AUTO_INCREMENT COMMENT 'ID',
  `strategy_name` varchar(32) CHARACTER SET utf8 COLLATE utf8_general_ci NOT NULL COMMENT '策略名称',
  `strategy_id` varchar(32) CHARACTER SET utf8 COLLATE utf8_general_ci NOT NULL COMMENT '策略ID',
  `strategy_type` varchar(32) CHARACTER SET utf8 COLLATE utf8_general_ci NOT NULL COMMENT '灵活-多种策略模式\r\n消息\r\n邮箱\r\n支付\r\n等等',
  `strategy_bean_id` varchar(255) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL COMMENT '策略执行beanid',
  PRIMARY KEY (`id`) USING BTREE
) ENGINE = InnoDB AUTO_INCREMENT = 11 CHARACTER SET = utf8 COLLATE = utf8_general_ci COMMENT = '策略' ROW_FORMAT = Dynamic;

-- ----------------------------
-- Records of strategy
-- ----------------------------
INSERT INTO `strategy` VALUES (1, 'kafka消息渠道', 'kafka', 'mq', 'kafkaDBStrategy');
INSERT INTO `strategy` VALUES (2, 'rabbitmq消息渠道', 'rabbit', 'mq', 'rabbitDBStrategy');
INSERT INTO `strategy` VALUES (3, 'rocketmq消息渠道', 'rouket', 'mq', 'rocketDBStrategy');
INSERT INTO `strategy` VALUES (4, '163邮箱渠道', '163', 'email', '163EmailStrategy');

SET FOREIGN_KEY_CHECKS = 1;

application.yml文件

###服务启动端口号
server:
  port: 8080

spring:
  datasource:
    url: jdbc:mysql://127.0.0.1:3306/strategy?useUnicode=true&characterEncoding=UTF-8&useSSL=true&serverTimezone=UTC
    username: root
    password: root
    driver-class-name: com.mysql.cj.jdbc.Driver


####打印MyBatias日志
logging:
  level:
    ### 开发环境使用DEBUG 生产环境info或者error
    com.xuyu.mapper: DEBUG

entity类

@Data
@TableName("strategy")
public class StrategyEntity {

    @TableId
    private Integer id;

    @TableField("strategy_name")
    private String strategyName;

    @TableField("strategy_id")
    private String strategyId;

    @TableField("strategy_type")
    private String strategyType;

    @TableField("strategy_bean_id")
    private String strategyBeanId;
}

定义抽象行为策略

public abstract class BehaviorStrategy {

    /**
     * 执行具体消息策略模式
     */
    protected abstract void specificMsgStrategy ();
}

mapper类

@Repository
public interface StrategyMapper {

    /**
     * 根据策略id和类型查询不同的策略
     *
     * @param strategyId 策略id
     * @param strategyType 策略类型
     * @return
     */
    @Select("SELECT id as id, strategy_name as strategyName, strategy_id as strategyId, strategy_type as strategyType, strategy_bean_id as strategyBeanId\n" +
            "FROM strategy s  WHERE s.strategy_id = #{strategyId} and strategy_type = #{strategyType}")
    StrategyEntity getStrategyByStrategyId(@Param("strategyId") String strategyId, @Param("strategyType") String strategyType);
}

kafka消息渠道

@Component
@Slf4j
public class KafkaDBStrategy extends BehaviorStrategy {

    @Override
    public void specificMsgStrategy() {
        log.info("执行kafkaMQ消息模式发送消息");
    }
}

rabbit消息渠道

@Component
@Slf4j
public class RabbitDBStrategy extends BehaviorStrategy {
    @Override
    public void specificMsgStrategy() {
        log.info("执行rabbitMQ消息模式发送消息");
    }
}

rocket消息渠道

@Component
@Slf4j
public class RocketDBStrategy extends BehaviorStrategy {

    @Override
    public void specificMsgStrategy() {
        log.info("执行rocketMQ消息模式发送消息");
    }
}

工厂模式,通过将策略具体执行者,注入到容器中,工厂通过beanid到容器中执行不同的策略具体执行者,相比第一种就简单很多

@Component
public class FactoryDBStrategy {

    @Autowired
    private StrategyMapper strategyMapper;

    public <T> T getStrategy(String strategyId, String type, Class<T> t) {
        if (StringUtils.isBlank(strategyId)) {
            return null;
        }
        if (StringUtils.isBlank(type)) {
            return null;
        }

        StrategyEntity strategyEntity = strategyMapper.getStrategyByStrategyId(strategyId, type);
        if (Objects.isNull(strategyEntity)) {
            return null;
        }

        T bean = SpringUtils.getBean(strategyEntity.getStrategyBeanId(), t);
        return bean;
    }

}

controller类

@RestController
public class StrategyDBController {

    @Autowired
    private FactoryDBStrategy factoryDBStrategy;

    @GetMapping("/getDBStrategy")
    public void getStrategy(@RequestParam("strategyId") String strategyId, @RequestParam("strategyType") String strategyType) {
        BehaviorStrategy strategy = factoryDBStrategy.getStrategy(strategyId, strategyType, BehaviorStrategy.class);
        strategy.specificMsgStrategy();
    }
}

效果图

访问:http://localhost:8080/getDBStrategy?strategyId=rouket&strategyType=mq

到此策略模式就结束啦

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,928评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,192评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,468评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,186评论 1 286
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,295评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,374评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,403评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,186评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,610评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,906评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,075评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,755评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,393评论 3 320
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,079评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,313评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,934评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,963评论 2 351