使用Spring+观察者模式处理绩效评定流程状态变更及邮件发送

   绩效评定流程在每月所有员工绩效计划打分结束后,由系统自动按评定关系表(树状结构,父结点是子结点的评定人,各部门总监为最底层评定人)发起任务,将最底层评定人下属的信息组装为表插入审批表单中,平行的最底层评定人为并行的子任务。最后一个平行最底层评定人评定结束后,系统会自动将新建子任务给平行最底层评定人的评定关系上层,并将所有平行最底层评定人的评定信息汇总作为审批表单,由此迭代直至找到平台级最终评定关系为止,流程结束。由于该流程极其特殊,既不是普通的并行或串行,也不是普通的子任务或会签,除发起结点、结束结点外,只有一个评定结点,所有评定人的评定活动均被看作评定结点的子任务,jbpm认为所有子任务均是并行关系,在业务逻辑中使用评定关系表等基础信息标识不同子任务的层级关系。由于无法使用统一的流程流转,在设计好流程及表单后,单独设计引擎驱动该流程,并在流程运转过程中更新基础信息表并负责发送邮件。

  单独的流程引擎设置、员工绩效基础信息表更新及邮件的发送由不同程序员开发。其中流程引擎在驱动流程运转过程中,会根据需要通知相应逻辑更新表及发送邮件。在开发引擎时,我使用观察者模式对流程进行重构使更新员工绩效基础信息和邮件发送逻辑的开发均独立进行,使代码的耦合进一步降低。

  在系统中存在这样的情况:一个功能点触发的动作会引起相关一个或者多个功能点在业务上进行对应的数据处理。而到底有几个功能点要做出相应,要看客户实施了哪些需要作出响应的具体功能点。

      例如:A功能点的某项业务会触发B、C、D三功能点做出回应,而客户订阅了哪些功能点是个未知数,也许客户不需要C功能点,对A功能点业务操作作出响应的仅有B、D。

      根据上面的需求可以得出,设计上要尽量的松散耦合,保持各功能点的独立性。观察者模式责无旁贷的跳了出来。

       我们的系统在整体上采用Spring framework来进行Bean管理。利用Spring通过配置文件来加载具体的类的方式——相当于被包装了的工厂,使得代码更加灵活。同样,观察者模式的应用可以很好的借用Spring framework提供的平台,变得更加灵活。

        首先,我们定义一个抽象的观察者

/**
 * TODO(描述类的职责)
 * @author chao.gao
 * @date 2015年11月30日 上午8:50:59
 * @version <b>1.0.0</b>
 */
public interface BaseObserverHandler extends Observer {

}
/**
 * A class can implement the <code>Observer</code> interface when it
 * wants to be informed of changes in observable objects.
 *
 * @author  Chris Warth
 * @version %I%, %G%
 * @see     java.util.Observable
 * @since   JDK1.0
 */
public interface Observer {
    /**
     * This method is called whenever the observed object is changed. An
     * application calls an <tt>Observable</tt> object's
     * <code>notifyObservers</code> method to have all the object's
     * observers notified of the change.
     *
     * @param   o     the observable object.
     * @param   arg   an argument passed to the <code>notifyObservers</code>
     *                 method.
     */
    void update(Observable o, Object arg);
}

  具体观察者:

public class EndProcessMailSentHandler implements BaseObserverHandler{
//实际update方法,具体实现略
}

   在观察者模式中,目标角色是被客户程序调用,通过目标角色来通知具体观察者角色。在传统观察者模式中,目标角色中维护有一个观察者角色的列表,列表中的观察者角色通过其它程序进行添加维护——这就注定了观察者列表的改变会引起代码的修改。而通过Spring framework的配置方式注册观察者角色则避免了这种情况。

/**
 * TODO(描述类的职责)
 * @author chao.gao
 * @date 2015年11月27日 下午4:15:36
 * @version <b>1.0.0</b>
 */
public abstract class BaseObservedHandler extends Observable {

	private List<BaseObserverHandler> observerHanlers;

	/**
	 * 被观察者进行操作
	 * @author chao.gao
	 * @date 2015年11月27日 下午4:18:56
	 * @param context
	 */
	public void obServedOpertion(EvaluationPorcessContext context) {
		if (observerHanlers == null || observerHanlers.isEmpty())
			return;
		for (BaseObserverHandler baseObserverHandler : observerHanlers) {
			super.addObserver(baseObserverHandler);
		}
		super.notifyObservers(context);
	}

	/**
	 * 获得observerHanlers
	 * @return List<BaseObserverHandler>
	 */
	public List<BaseObserverHandler> getObserverHanlers() {
		return observerHanlers;
	}

	/**
	 * 设置observerHanlers
	 * @param observerHanlers
	 */
	public void setObserverHanlers(List<BaseObserverHandler> observerHanlers) {
		this.observerHanlers = observerHanlers;
	}

}

  下面是Spring framework配置文件的一个片断——它应该出现在用到目标角色的具体客户程序对应的Spring配置文件中。它将具体观察者的别名存放在一个list里面,使用Spring为生成的目标角色注入。

  如下:

<bean 
		class="com.fx.oa.module.per.evaluation.workflow.server.service.impl.EndProcessService">
		<property name="jbpmOperatorService" ref="jbpmOperatorService" />
		<property name="evaluationProcessService" ref="evaluationProcessService" />
		<property name="observerHanlers">
			<list>
				<ref bean="endProcessHandler"/>
				<ref bean="endProcessMailSentHandler" />
				<ref bean="endProcessSaveFormHandler"/>
			</list>
		</property>
	</bean>
更多相关文章
  • Sql2005自动备份并邮件通知状态之三设置邮件通知
    Sql2005自动备份并邮件通知状态之三设置邮件通知本章节为最后一章节,介绍如何创建代理操作员和设置备份的邮件通知创建操作员和设置邮件通知单击创建操作员,输入操作员名称和邮件地址单击确定即可.下面我们可以看到代理作业里面产生了一个以数据库维护计划为前缀的子代理作业,我们编辑它即可右键单击作业,选择属 ...
  • 转载自使用Spring的JAVA Mail支持简化邮件发送 闲来无事,翻看<Spring in Action>,发现Spring集成了对JAVA Mail的支持,有点小激动的看了一遍,嗯,话说真的简单了很多. Spring的邮件发送的核心是MailSender接口,在Spring3.0中 ...
  • 使用spring提供的MailSender和JavaMailSender类. 1.邮件对象类 package cn.luxh.app.mail; import java.util.List; import org.springframework.core.io.AbstractResource; p ...
  • Spring邮件抽象层的主要包为org.springframework.mail.它包括了发送电子邮件的主要接口MailSender,和值对象SimpleMailMessage,它封装了简单邮件的属性如from, to,cc, subject,text. 包里还包含一棵以MailException为 ...
  • 撰写人:度量科技http://www.delit.cn
  • C#实现对站点、程序池状态的监控,以及URL能正常返回的监控,状态异常,邮件预警
    需求:自动化组提出需要,要对IIS上的站点进行监控,异常停止后报警需求分析:这站点的运行正常需要多方面的监控,如站点,程序池,资源,所以针对这需求做了三方面的监控.站点状态的监控站点对应的程序池的监控URL的监控,监控url能返回200的状态码数据库设计:页面展示:配置list页配置Detail页预 ...
  • 项目中经常用到邮件功能,在这里简单的做一下笔记,方便日后温习. 首先需要在配置文件jdbc.properties添加: #------------ Mail ------------); javaMailSender.setUsername("[email protected]" ...
  • - (void)initNetworkMonitor {     NSURL *baseURL = [NSURL URLWithString:@"http://www.baidu.com/"];     AFHTTPRequestOperationManager *manager ...
一周排行
  •     先说一下我的观点,以前测试过Vmware虚拟桌面产品VDI,总的来说就是在ESXServer上建立很多个Winxp的虚拟机,客户机上通过远程桌面或者web就可以使用这些虚拟桌面(ESXserver上的N个wi ...
  • 最近因为一个监控相关的项目,深入研究了一下 windows 的 远程桌面的相关知识. 1. 如何让关闭了远程桌面连接的用户,对应的 session 立即退出 windows server. 大家使用 mstsc.ex ...
  • 跟互联力量学Asp.netMVC3-传值
    本节重点说明如何将控制器中的数据传递给视图的问题.如果对之前的内容不清楚,请参考<跟 ...
  • CronTrigger CronTriggers往往比SimpleTrigger更有用,如果您需要基于日历的概念,而非SimpleTrigger完全指定的时间间隔,复发的发射工作的时间表.CronTrigger,你可 ...
  • <!DOCTYPE>标签必须放在整个html文档的第一行,之后一行就是从<html>标签开始,所有浏览器都支持<!DOCTYPE>标签. <!DOCTYPE>标签是用来 ...
  •     不少朋友需要在DedeCms的问答系统.留言本.会员空间导航里调用系统标签,但默认是不支持系统标签调用的,需要对系统文件进行简单的修改.      第一步:打开"/include/common.fu ...
  • ==本文会是"选项内容"的最后一期讲解,主要会讲讲-w和-r两个选项.tcpdump的选项很多,多达50个,其他我没有涉及的选项,还是要大家自己通过man tcpdump的方式来学习了.实在研究不 ...
  • OSPF系列小实验之8:路由汇总
    OSPF下的汇总分为区域汇总和外部路由汇总,与rip或者eigrp不同的是,OSPF下的汇 ...
  •     sass的语法运用     1.语法    sass有两种后缀名文件:一种后缀名为sass,不使用大括号和分号:另一种就是我们这里使用的scss文件,这种和我们平时写的css文件格式差不多,使用大括号和分号. ...
  • 在进行这些习题的解答的时候往往要注意ip的分类,即分为A类.B类.C类,但是其解题方法都是差不多的,只是几类当中的所占有的八位的个数不一样,A类的网络地址是第一个八位中的某些为是on状态,后面的全部为off状态:B类 ...