读者可以在eclipse中导入附件的项目,执行main.java体验"步骤执行容器"的效果(温馨提示,stepframework依赖了dom4j,在附件中的dependence目录含有该lib)。
问题背景与实现简述
通常情况下,一项任务可以分为多个步骤,每个步骤之下又能分为几个子步骤。最简单的实现方法就是:使用一个主类调用几个步骤方法去完成任务;每个步骤方法执行的时候,能够读取的参数都是同一配置文件下的所有参数。然而,到了后期,任务的需求变得越来越灵活和复杂,前面实现方法就存在以下缺点了:
1 不能做到某些步骤的配置式替换、去掉或者添加;
2 当步骤越来越多的时候,参数命名容易冲突,通常需要添加足够的前缀。况且一个步骤能到其他所有的步骤的参数,也不太合理。
3 参数在步骤间传递的是个设计难题。通常来说,在多个方法或者类之间传递某个值,要么需要使用一个static变量、threadlocal之类的全局变量来维护,要么在方法中传递参数。前者会一定程度上增加维护的难度(需要清楚某个参数何时放进去,何时拿出来,需要知道全局变量里面究竟会有哪些参数),而后者会让代码看起来好臃肿(参数太多了)而且耦合性较强。
4 当步骤的数量庞大起来的时候,让人很难理清其中的执行的顺序和逻辑。
例如做一个增量补丁制作工具,刚开始按照 ”SVN-DIFF-->SVN下载-->全量编译SRC-->增量编译SRC-->复制文件-->整合sql执行脚本文件“的思路做了一个简单的工具。后面发现,有时候需要SVN下载后停顿一下,以便手工删除某些文件;整合sql执行脚本文件针对不同的项目需要替换;后面有人提议希望能够直接只指定某几个步骤,而不是每次都从头到尾执行,太浪费时间了,例如只需要执行整合sql脚本等等。这时候,原来简单的一个主类调用多个步骤方法的程序很难使用。
重构:采用配置文件、责任链模式、聚合模式解决以上问题
1 使用step.xml配置文件描述所有的步骤组件、步骤参数和要执行的步骤顺序。
通过配置该文件,可以灵活地替换、去掉或者添加组件。同时,该文件,能够非常清晰地描述整个任务的执行的过程。
<main>add,multiply,divideAndSubtractionStepSuite</main> <all-step> <step name="add" class="test.Add" params-ref="add"/> <step name="multiply" class="test.Multiply" params-ref="multiply"/> <step name="divideAndSubtractionStepSuite" class="test.TryStepSuit" params-ref="divideAndSubtractionStepSuite" /> </all-step>
看了这个文件,即使我不说明,都大概能猜到该程序要依次运行add,multiply,divideAndSubtractionStepSuite这三个步骤,步骤对应的类和参数在<step/>中得到进一步的描述。
细节1
divideAndSubtractionStepSuite
这个步骤其实集合了除法步骤和减法步骤,融合为一步骤(具体参见附件或实例代码)。这个特性是为了满足可读性的要求。例如一个增量编译,包含了下载增量文件,编译,COPY文件三个子步骤,融合为一个名为compileDiff的步骤,显然好读,更易理解。
细节2
配置文件没有考虑StepSuite的配置项,只有step配置项。
stepframwork将包含多个step和stepSuite组件的集合称为stepSuite。目前对StepSuite下的各个组件共用一个参数集合,因为考虑到一个StepSuite,大多数情况下是不应该存在重复参数的,而且,我认为一个StepSuite作为一个namespace已经足够了,细化到Step似乎有些多余。万一如果一个StepSuite下真存在重复组件和重复参数名,实际上很可能应该将一个StepSuite分拆为两个。另一方面,考虑到如果要使每个StepSuite组件下的所有Step单独使用自己的参数集合,配置将变得复杂,后面的人维护起来也麻烦。stepframwork并不是不能实现每个Step对应一个参数集合,而是为了维护和易学,stepframework做了一个“不完美”的选择, 因此配置文件中没有类似
<stepSuite name="divideAndSub">
<step step-ref="divide"/>
<step step-ref="sub"/>
</stepSuite>
的配置。但是可以通过写一个TestSuite的子类实现同等效果,也很简单。
例如:
package test; import stepframework.StepSuite; public class TryStepSuit extends StepSuite{ public TryStepSuit(){ this.addStep(new Divide()).addStep(new Sub()); } }
2 每个步骤组件关联一个参数集合,当参数集合中查找不到想要的参数,则使用责任链在前面步骤组件关联的参数集合中查找参数。
这样设计是因为通常情况下,后面执行的步骤需要前面步骤的某些参数,而责任链能非常优雅的实现这一点。责任链还提供其他两个好处:
a)免去了重用前面的某个步骤组件时需要在参数名前面添加前缀的问题,因为即使重名,责任链保证首先会读到该组件实例关联的参数。
b)利用责任链,可以把前一步骤的执行结果,传递到下一个步骤。
此外,每个步骤组件关系的参数集合,都是由stepframework自动注入到具体的步骤组件对象中,无需步骤组件开发人员处理。
题外话:在现在使用责任链来实现参数的动态查找之前,我是这样设计的,一个步骤组件关联一个参数集合,该集合包含了该步骤所需的所有参数,但是这些参数与其他步骤的参数有不少重复。对于一个有代码洁癖的程序原来说,这是不能容忍的事情,苦思一番之后,灵感乍现,javascript原型链不就是解决我这个问题的最佳实践吗?于是,责任链的实现诞生了。
<all-params> <params name="add"> <param name="left" value="10"/> <param name="right" value="20"/> <param name="result" value=""/> </params> <params name="multiply"> <param name="multi_num" value="3"/> </params> <params name="divideAndSubtractionStepSuite"> <param name="div_num" value="2"/> <param name="sub_num" value="3"/> </params> </all-params>
package stepframework; import java.util.Map; public class Params { private Map<String,String> map; private Params parent; public Params(Map<String, String> map, Params parent) { super(); this.map = map; this.parent = parent; } public void put(String paramName, String paramValue){ map.put(paramName, paramValue); } //责任链查找 public String get(String paramName){ String paramValue = map.get(paramName); if(null != paramValue){ return paramValue; }else if(null != parent){ return parent.get(paramName); }else{ return ""; } } }
3 对于步骤集合(StepSuite)和单一步骤(Step),使用了聚合模式,统一作为一个IStep的实现,容器通过该方式,统一处理所有的步骤,而无视所谓步骤与子步骤,程序变得更简单。
package stepframework; import java.util.ArrayList; import java.util.List; public class StepSuite implements IStep{ private Params params; private List<IStep> steps = new ArrayList<IStep>(); public StepSuite(){ } public StepSuite addStep(IStep step){ steps.add(step); return this; } //聚合模式的关键,集合对象,依次执行每个子对象 @Override public Result run() { for(IStep step : steps){ step.setParams(params); Result rs = step.run(); if(!rs.isSuccess()){ return rs; } } return Result.SUCCESS; } public void setParams(Params params){ this.params = params; } public void putParam(String paramName, String paramValue){ params.put(paramName, paramValue); } public String getParam(String paramName){ return params.get(paramName); } }
StepFramework使用示例
stepframework使用非常简单,只需两步:1. 扩展基类创建步骤组件,2. step.xml中描述如何运行这些组件
下面演示了加减乘除4个步骤的依次执行,每个步骤执行的结果,都能传递到后一步骤。
1. 编写步骤类,继承Step,或者StepSuite(包含多个Step或者StepSuite)。
下面定义了加减乘除4个Step,和1个综合了“除和减”两个Step的集合StepSuite。
对于单一步骤Step,需要继承Step类,重写run()方法,编写步骤的具体逻辑。在run方法中使用getParam查找参数,使用putParam设置传递到后面步骤的结果参数。如果执行失败,创建一个Result对象,填充错误信息 ,并且stepframework会马上体制,如果执行成功,直接返回Result.SUCCESS即可。
对于Step或StepSuite的步骤集合StepSuite,需要继承StepSuite类,编写构造函数,调用add()方法添加所需的Step或StepSuite。这些Step和StepSuite都共同关联同一个参数结合params。
package test; import stepframework.Result; import stepframework.Step; public class Add extends Step{ @Override public Result run() { String left = this.getParam("left"); String right = this.getParam("right"); Integer i = Integer.valueOf(left) + Integer.valueOf(right); String result = String.valueOf(i); this.putParam("result",result ); System.out.println(left+"+"+right+"="+result); return Result.SUCCESS; } }
package test; import stepframework.Result; import stepframework.Step; public class Multiply extends Step{ @Override public Result run() { String result = this.getParam("result"); String multi_num = this.getParam("multi_num"); String orginResult = result; result = String.valueOf(Integer.valueOf(result)*Integer.valueOf(multi_num)); System.out.println(orginResult+"*"+multi_num+"="+result); this.putParam("result", result); return Result.SUCCESS; } }
package test; import stepframework.Result; import stepframework.Step; public class Divide extends Step{ @Override public Result run() { String result = this.getParam("result"); String divNum = this.getParam("div_num"); String orginResult = result; result = String.valueOf(Integer.valueOf(result)/Integer.valueOf(divNum)); System.out.println(orginResult+"/"+divNum+"="+result); this.putParam("result", result); return Result.SUCCESS; } }
package test; import stepframework.Result; import stepframework.Step; public class Sub extends Step{ @Override public Result run() { String result = this.getParam("result"); String sub_num = this.getParam("sub_num"); String orginResult = result; result = String.valueOf(Integer.valueOf(result) - Integer.valueOf(sub_num)); System.out.println(orginResult+"-"+sub_num+"="+result); this.putParam("result", result); return Result.SUCCESS; } }
package test; import stepframework.StepSuite; public class TryStepSuit extends StepSuite{ public TryStepSuit(){ this.addStep(new Divide()).addStep(new Sub()); } }
2. 配置step.xml,让stepframework自动按顺序执行各个步骤。
a)在step.xml中的<step/>中,添加上步骤名,步骤实现类和引用的参数集合名。
b)在step.xml中的<params/>中,配置每个步骤所需要的参数。
c)在step.xml中main中,配置要执行的步骤,stepframework会按照先后顺序执行。
d)step.xml放在classpath下
<step-root> <main>add,multiply,divideAndSubtractionStepSuite</main> <all-step> <step name="add" class="test.Add" params-ref="add"/> <step name="multiply" class="test.Multiply" params-ref="multiply"/> <step name="divideAndSubtractionStepSuite" class="test.TryStepSuit" params-ref="divideAndSubtractionStepSuite" /> </all-step> <all-params> <params name="add"> <param name="left" value="10"/> <param name="right" value="20"/> <param name="result" value=""/> </params> <params name="multiply"> <param name="multi_num" value="3"/> </params> <params name="divideAndSubtractionStepSuite"> <param name="div_num" value="2"/> <param name="sub_num" value="3"/> </params> </all-params> </step-root>
3 调用StepRunner类执行程序
public class Main { public static void main(String[] args) throws URISyntaxException, DocumentException, ClassNotFoundException, InstantiationException, IllegalAccessException{ StepRunner.run(); } }
相关推荐
标题中的“Kettle实现步骤循环执行”涉及到的是数据集成工具Pentaho Data Integration(简称Kettle或PDI)的一种高级用法,即如何在工作流(Job)或转换(Transformation)中实现步骤的循环执行。Kettle是一款强大的...
本篇将深入探讨Servlet的基础知识及其运行环境——Servlet容器模型。 首先,让我们了解Servlet的基础。Servlet是一个Java类,遵循javax.servlet.Servlet接口,它允许开发者扩展服务器的功能。当用户向Web服务器发送...
### Qt5.9.1可执行程序转换为安装文件的步骤详解 #### 一、简述 本篇文章将详细介绍如何将Qt5.9.1开发的可执行程序转化为可直接安装使用的`.exe`安装文件。这不仅适用于Qt5.9.1版本,也适用于其他版本的Qt框架开发...
《压力容器设计步骤详解》 压力容器是一种在工业生产中广泛应用的重要设备,它主要用于存储、运输或处理具有高压、高温的气体或液体。其设计过程严谨且复杂,涉及到材料选择、强度计算、工艺流程等多个方面。以下是...
# 在容器中执行bash命令 docker exec -it id /bin/bash 3.安装openssh-server并启动 apt-get update apt-get install openssh-server # 启动之前需手动创建/var/run/sshd,不然启动sshd的时候会报错
编写Dockerfile文件构建镜像,启动容器。
《压力容器的设计步骤》 压力容器的设计是一项严谨且复杂的工作,涉及到多个方面,包括设备参数的确定、压力容器类型的划分、设计参数的选择以及容器的基本尺寸和配置等。下面将详细阐述这些步骤。 首先,压力容器...
此外,对于涉及多个步骤的复合操作,如添加和删除元素,同步容器无法保证线程安全性。 例如,`Vector`的`deleteVector`方法可能会遇到并发问题,因为不同线程可以交错执行`size()`和`remove()`,导致异常。为了解决...
下面是一个简化的步骤来创建带有慢动画效果的折叠容器: 1. **创建自定义控件类**:首先,创建一个新的用户控件(UserControl)继承自`Panel`。这将是我们的折叠容器。 2. **设置初始状态**:在控件的构造函数中,...
容器云搭建3master操作步骤
下面将详细介绍 Kettle 部署和定时作业执行的步骤。 一、java 安装和配置 在部署 Kettle 之前,需要安装和配置 java 环境。首先,需要查看系统是否已经安装了 JDK,可以使用命令“java -version”来检查。如果没有...
耐压试验则是验证容器强度的重要步骤。 三、压力容器的安全附件是保障设备正常运行的关键。常见的安全附件包括压力表、安全阀、爆破片、液位计、温度计等。这些附件需要定期校验和维护,确保其在紧急情况下能正确...
容器化技术,尤其是Docker容器的出现,为微服务架构的实现提供了便利。而Docker Compose,作为Docker官方的容器编排工具,进一步简化了容器化应用的部署和管理流程。 在本地开发部署架构和需求部分,文档提到了本地...
配置和使用Docker涉及几个关键方面,包括安装、创建镜像、运行容器以及管理容器等。以下是基本的Docker配置和使用步骤: ### 1. 安装Docker 首先,你需要在你的操作系统上安装Docker。Docker支持多个操作系统,...
2. 数据处理自动化:使用宏执行器,可以录制一系列数据处理步骤,然后自动执行这些步骤,从而实现自动化数据处理。 3. 文件管理自动化:使用宏执行器,可以录制一系列文件管理步骤,然后自动执行这些步骤,从而实现...
控制部分主要由伺服放大器、手操器和电动执行机构组成,它们按照PLC发出的指令,对进料阀和排空阀等执行元件进行操作,以实现对压力容器内参数的精确控制。 上位机程序方面,本系统选择了Realinfo组态软件,它能够...
容器 runtime 负责容器的执行环境,提供了容器的网络、存储和安全性等功能。容器网络是容器服务的基本组件,提供了容器之间的网络通信功能。存储管理是容器服务的重要组件,负责容器的存储管理和数据存储。监控与...
制造过程中可能包括热处理、焊接、表面处理、检验和试验等步骤。检验和试验通常包括无损检测、压力试验等,以确保容器的安全性和可靠性。 知识点六:化工压力容器维护与管理 化工压力容器在使用过程中需要定期进行...
5. **焊接工艺**:焊接是压力容器制造中的关键环节,涉及坡口制备、焊接方法(如手工电弧焊、氩弧焊等)、焊接参数选择、焊接顺序等,必须制定并执行焊接工艺规程。 6. **无损检测**:包括超声波检测、射线检测、...
LoadRunner 压力测试简单用户登录操作步骤详解 LoadRunner 是一款功能强大且流行的性能测试工具,广泛应用于网站压力测试、负载测试、性能优化等领域。今天,我们将详细介绍 LoadRunner 中的压力测试简单用户登录...