流程编排最佳实践(java版本)

背景

为解耦复杂逻辑而生,帮助解耦业务代码,让每一个业务片段都是一个组件.支持规则文件的热加载,即时完成修改生效.

类图概览

各类关系

设计原则

一.开闭原则

  对外扩展开发,对内修改关闭

二.单一职责原则

  解释:一个类只负责一项职责。
  实现:这里的职责代表一次编排,例如SequenceTicketStageExecutorFlow 代表此次编排的内容,如果符合可以复用,不符合增加新类编排内容。

三.依赖倒置原则

解释:
    1.上层模块不应该依赖底层模块,他们都应该依赖于抽象
    2.抽象不应该依赖于细节,细节应该依赖于抽象
实现:执行流程面向接口(TicketExecutorFlow)。节点处理面向接口(TicketStage)。

四.里氏替换原则

解释:子类不应该覆盖覆盖父类的方法
实现:SequenceTicketStageExecutorFlow 没有覆盖 TicketStageExecutorFlow的execute方法,覆盖的方法在父类中都是抽象方法。

五.最少知道原则

解释:降低类之间的耦合性,提高代码的复用性

使用指南

xml配置形式

flowConfig.xml 总配置,配置多种类型编排逻辑

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd">

    <import resource="classpath:/flow/TicketSequenceConfig.xml" />
</beans>

TicketSequenceConfig.xml 具体编排内容

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:context="http://www.springframework.org/schema/context"
       xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd">
    <import resource="classpath*:/common/beans.xml"/>
    <bean id="ticketDetailExecutorFlow" class="com.idanchuang.customer.satisfaction.data.manager.flow.SequenceTicketStageExecutorFlow">
        <property name="bizCheckStageList">
            <list>
                <ref bean="ticketDetailParamsCheckStage" />
            </list>
        </property>
    </bean>

注解配置方式

flowConfig.properties 配置属性
#demo
ticket.detail.get.detailCheckList = ticketDetailParamsCheckStage
ticket.detail.get.detailBizList = ticketDetailApplyInfoStage,ticketDetailAddressStage

config/TicketGetConfig.java 配置代码
@Configuration
@PropertySource(value={"classpath:/flowConfig.properties"})
public class TicketGetConfig {

    @Value("#{'${ticket.detail.get.detailCheckList}'.split(',')}")
    private List<String> detailCheckList;

    @Value("#{'${ticket.detail.get.detailBizList}'.split(',')}")
    private List<String> detailBizList;

    @Resource
    private ApplicationContext applicationContext;


    @Bean(name = "ticketDetail")
    public SequenceTicketStageExecutorFlow ticketDetail() {
        SequenceTicketStageExecutorFlow sequenceTicketStageExecutorFlow = new SequenceTicketStageExecutorFlow();
        sequenceTicketStageExecutorFlow.setBizCheckStageList(convertTicketStage(detailCheckList));
        sequenceTicketStageExecutorFlow.setResultStageList(convertTicketStage(detailBizList));
        return sequenceTicketStageExecutorFlow;
    }

    @Bean(name = "ticketBriefDetail")
    public SequenceTicketStageExecutorFlow ticketBriefDetail() {
        SequenceTicketStageExecutorFlow sequenceTicketStageExecutorFlow = new SequenceTicketStageExecutorFlow();
        sequenceTicketStageExecutorFlow.setResultStageList(convertTicketStage(detailBizList));
        return sequenceTicketStageExecutorFlow;
    }

    private List<TicketStage> convertTicketStage(List<String> ticketStage) {
        List<TicketStage> result = new ArrayList<>();
        if (CollectionUtils.isEmpty(ticketStage)) {
            return result;
        }
        ticketStage.forEach(stageName -> {
            TicketStage tmpStage = applicationContext.getBean(stageName, TicketStage.class);
            result.add(tmpStage);
        });
        return result;
    }

}

单元测试

@RunWith(SpringRunner.class)
@SpringBootTest(classes = DataApplication.class)

public class SequenceTest {

    @Resource(name = "ticketDetail")
    private TicketStageExecutorFlow<TicketDetailFlowParam, TicketeDetailFlowResult, TicketDetailFlowContext> ticketDetail;

    @Resource(name = "ticketBriefDetail")
    private TicketStageExecutorFlow<TicketDetailFlowParam, TicketeDetailFlowResult, TicketDetailFlowContext> ticketBriefDetail;

    @Resource(name = "ticketDetailExecutorFlow")
    private TicketStageExecutorFlow<TicketDetailFlowParam, TicketeDetailFlowResult, TicketDetailFlowContext> ticketDetailExecutorFlow;


    @Test
    public void testA() {
        TicketDetailFlowParam ticketDetailFlowParam = new TicketDetailFlowParam();
        ticketDetailFlowParam.setId(12L);

        ticketDetail.execute(ticketDetailFlowParam);
        ticketBriefDetail.execute(ticketDetailFlowParam);
        ticketDetailExecutorFlow.execute(ticketDetailFlowParam);
    }

}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 第三部分 软件工程 考试题型:概念问答题、实践案例题总分:50分 一、软件过程 软件过程的概念 软件过程是指软件整...
    小K哥哥阅读 2,512评论 0 2
  • 三重:代码、底层内存、源码 第一阶段:开发常用JavaSE基础、IDE、Maven、Gradle、SVN、Git、...
    guodd369阅读 17,003评论 1 44
  • Python6大设计原则 阅读目录 内容总览 六大设计原则都有哪些 一、单一职责原则 二、里氏替换原则 三、依赖倒...
    tomtiddler阅读 1,575评论 0 0
  • 1 设计模式的定义 设计模式是一套编码理论,由软件的先辈们总结出的一套可以反复使用的经验——代码组织的方法论。设计...
    Jerry_1116阅读 737评论 0 1
  • 本文由 玉刚说写作平台 提供写作赞助原作者:却把清梅嗅 原文地址:https://mp.weixin.qq.co...
    却把清梅嗅阅读 2,015评论 2 10