flowable中命令模式的应用

flowable中大量使用了设计模式中的 命令模式,咱来看看源码.

首先出场的执行器接口:CommandExecutor 只有一个实现类CommandExecutorImpl 

负责执行所有的命令.执行器有个最重要的方法

    /**
     * Execute a command with the default {@link CommandConfig}.
     */
    <T> T execute(Command<T> command);

这个执行器在创建流程引擎的时候执行器相关的配置被初始化,对象会被实例化.AbstractEngineConfiguration

public void initCommandExecutors() {
    initDefaultCommandConfig();
    initSchemaCommandConfig();
    initCommandInvoker();
    initCommandInterceptors();
    initCommandExecutor();
}
    public void initCommandExecutor() {
        if (commandExecutor == null) {
            CommandInterceptor first = initInterceptorChain(commandInterceptors);
            commandExecutor = new CommandExecutorImpl(getDefaultCommandConfig(), first);
        }
    }

执行器需要传入命令对象,以启动流程为例:

我们启动流程通常是按key启动,如runtimeService.startProcessInstanceByKey(procDefKey, businessTable + ":" + businessId, vars);

我们看源码是创建了一个StartProcessInstanceCmd的命令给执行器执行.

    @Override
    public ProcessInstance startProcessInstanceByKey(String processDefinitionKey, String businessKey, Map<String, Object> variables) {
        return commandExecutor.execute(new StartProcessInstanceCmd<ProcessInstance>(processDefinitionKey, null, businessKey, variables));
    }
StartProcessInstanceCmd<T> 实现了接口Command<ProcessInstance>,并且实现了命令的 execute方法
T execute(CommandContext commandContext);

这个StartProcessInstanceCmd的execute方法是真正的创建流程实例的逻辑.我们看到创建实例有两个逻辑,

1.获取ProcessDefinition

2.调用

processInstanceHelper.createProcessInstance()的方法创建实例并启动.

上面我们已经知道初始化了CommandExecutor,在此之前还初始化了默认的CommandInvoker对象.CommandInvoker 继承了AbstractCommandInterceptor

public void initCommandInvoker() {
    if (commandInvoker == null) {
        commandInvoker = new DefaultCommandInvoker();
    }
}

这个CommandInvoker就是执行command的真正执行者.CommandInvoker还用了责任链模式来处理日志和事务等通用任务.

LogInterceptor extends AbstractCommandInterceptor
SpringTransactionInterceptor extends AbstractCommandInterceptor
CommandContextInterceptor extends AbstractCommandInterceptor
CommandInvoker extends AbstractCommandInterceptor

CommandInvoker也是继承了AbstractCommandInterceptor的一个拦截器.抽象类

AbstractCommandInterceptor 实现了CommandInterceptor接口

拦截器在execute方法中执行了真正需要执行的Command的execute方法.

            // Execute the command.
            // This will produce operations that will be put on the agenda.
            agenda.planOperation(new Runnable() {
    
                @Override
                public void run() {
                    commandContext.setResult(command.execute(commandContext));
                }
            });
    
            // Run loop for agenda
            executeOperations(commandContext);

在这里flowable用了Agenda去轮询得到处理结果.

 

总结:

命令模式的三大件

CommandExecutor

Command

CommandInvoker

创建Command,交给执行器CommandExecutor,执行器的CommandInvoker最终去执行Command的execute方法的真正逻辑.

缺点:Command类可能会很多.

 

 

 

 

 

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

飞翔的咩咩

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值
OSZAR »