Spring+Mysql+Jbpm整合

前言

最近因为工作需要做一个工作流相关的DEMO,研究了一下JBPM,记录一下个人的心得与体会。

软件环境:

  • spring2.0.2
  • hibernate3.2.2
  • spring modules 0.8 (Jbpm3.1)
  • jbpm3.1.4
  • struts2.0.6

配置

Spring Module Jbpm模块提供了几个工具类用来整合spring和jbpm,关于具体的配置可以参见spring module下载包中的参考手册,按照上面的指示来就OK了,这里粘贴示例配置。

xml 代码
  1. xml version="1.0" encoding="UTF-8"?>  
  2. >  
  3. <beans default-autowire="byName" default-lazy-init="true">  
  4.     <bean id="approveWorkflow"  
  5.         class="org.springmodules.workflow.jbpm31.definition.ProcessDefinitionFactoryBean">  
  6.         <property name="definitionLocation"  
  7.             value="classpath:jbpm/audit/processdefinition.xml" />  
  8.     bean>  
  9.     <bean id="jbpmConfiguration"  
  10.         class="org.springmodules.workflow.jbpm31.LocalJbpmConfigurationFactoryBean">  
  11.         <property name="sessionFactory" ref="sessionFactory" />  
  12.         <property name="configuration" value="classpath:jbpm/jbpm.cfg.xml" />  
  13.         <property name="processDefinitions">  
  14.             <list>  
  15.                 <ref local="approveWorkflow" />  
  16.             list>  
  17.         property>  
  18.   
  19.     bean>  
  20.     <bean id="jbpmTemplate"  
  21.         class="org.springmodules.workflow.jbpm31.JbpmTemplate">  
  22.         <constructor-arg index="0" ref="jbpmConfiguration" />  
  23.         <constructor-arg index="1" ref="approveWorkflow" />  
  24.     bean>  
  25. beans>  
  26.   

比较关键的是为了能够使JBPM实体和业务实体使用同一个会话工厂,这样可以在JBPM流程实例中持久化业务实体对象。必须将业务实体映射和JBPM的实体映射进行整合。我的作法是改写JBPM本身提供的hibernate.cfg.xml文件,将业务实体包括在里面。

xml 代码
  1. xml version='1.0' encoding='utf-8'?>  
  2.   
  3.           "-//Hibernate/Hibernate Configuration DTD 3.0//EN"   
  4.           "http://hibernate.sourceforge.net/hibernate-configuration-3.0.dtd">  
  5.   
  6. <hibernate-configuration>  
  7.   <session-factory>     
  8.   
  9.         <property name="hibernate.cache.use_second_level_cache">  
  10.             false   
  11.         property>          
  12.         <property name="hibernate.cache.use_query_cache">  
  13.             false   
  14.         property>  
  15.         <property name="hibernate.dialect">org.hibernate.dialect.MySQL5Dialectproperty>  
  16.         <property name="hibernate.show_sql">falseproperty>  
  17.         <property name="hibernate.query.factory_class">  
  18.             org.hibernate.hql.ast.ASTQueryTranslatorFactory   
  19.         property>  
  20.            
  21.            
  22.         
  23.     <mapping resource="com/emap/jbpm/model/Apply.hbm.xml"/>  
  24.   
  25.       
  26.     <mapping resource="org/jbpm/graph/action/Script.hbm.xml"/>  
  27.   
  28.     <mapping resource="org/jbpm/db/hibernate.queries.hbm.xml" />  
  29.        
  30.       
  31.     <mapping resource="org/jbpm/graph/def/ProcessDefinition.hbm.xml"/>  
  32.     <mapping resource="org/jbpm/graph/def/Node.hbm.xml"/>  
  33.     <mapping resource="org/jbpm/graph/def/Transition.hbm.xml"/>  
  34.     <mapping resource="org/jbpm/graph/def/Event.hbm.xml"/>  
  35.     <mapping resource="org/jbpm/graph/def/Action.hbm.xml"/>  
  36.     <mapping resource="org/jbpm/graph/def/SuperState.hbm.xml"/>  
  37.     <mapping resource="org/jbpm/graph/def/ExceptionHandler.hbm.xml"/>  
  38.     <mapping resource="org/jbpm/instantiation/Delegation.hbm.xml"/>  
  39.        
  40.     ........   
  41.        
  42.   session-factory>  
  43. hibernate-configuration>  

下面我们看看sessionFactory工厂的配置。

xml 代码
  1. xml version="1.0" encoding="UTF-8"?>  
  2. >  
  3. <beans default-autowire="byName" default-lazy-init="true">  
  4.   
  5.       
  6.     <bean id="dataSource" class="org.apache.commons.dbcp.BasicDataSource" destroy-method="close">  
  7.         <property name="driverClassName" value="${jdbc.driverClassName}"/>  
  8.         <property name="url" value="${jdbc.url}"/>  
  9.         <property name="username" value="${jdbc.username}"/>  
  10.         <property name="password" value="${jdbc.password}"/>  
  11.     bean>  
  12.        
  13.       
  14.     <bean id="sessionFactory"  
  15.         class="org.springframework.orm.hibernate3.LocalSessionFactoryBean">  
  16.         <property name="dataSource" ref="dataSource" />  
  17.         <property name="configLocations">  
  18.             <list>  
  19.                 <value>classpath:jbpm/hibernate.cfg.xmlvalue>  
  20.             list>  
  21.         property>  
  22.     bean>  
  23.       
  24.     <bean id="transactionManager" class="org.springframework.orm.hibernate3.HibernateTransactionManager">  
  25.         <property name="sessionFactory" ref="sessionFactory"/>  
  26.     bean>  
  27. beans>  

遗留问题

如何使用JBPM Process Designer插件?

       一直没找到如何使用JBPM Process Designer插件的使用文档,比如如何配置JBPM安装路径,如何部署。目前唯一用到的功能就是编写流程文件。

如何发布流程文件?

        对于如何发布流程文件,我比较同意如下帖子中的观点,编程实现或许是最简洁的方式。

www.pcdog.com/edu/java/2006/11/v171946.html

如何关联业务实体和流程实例?

        JBPM主要用来管理业务流程,记录每个流程进入哪个环节,同时还要保存一些状态,这些状态信息可能来自于业务实体。JBPM的实现方式是将这些状态信息序列化到数据库的表列。

        假定有一个订单处理的流程,现在要获取某个角色当前的所有任务列表,同时将关联的订单信息展示给用户,我们应该如何处理?目前我想到有以下几种方式:

        方式一:在构建任务实例的时候,将业务实体持久化到contextInstance,在获取任务列表时从任务实例中直接解析出业务实体。如果需要保存的业务实体数据量很大,这会给JBPM数据库造成很大的数据冗余。

        方式二:在构建任务实例的时候,仅将业务实体的唯一标识符持久到contextInstance,在获取任务列表时从任务实例中解析出任务实体的唯一标识符,然后再根据此标识符查询业务实体数据库。这种情况会造成查询一个包含N个的任务列表时,需要N+1次数据库查询,显然性能是无法满足需求的。

        方式三:是否可以在构建业务实体时,和TaskInstance进行关联?这种方式会造成业务实体和JBPM紧耦合,而且必须对JBPM本身有比较深刻的理解。

        有什么更好的办法解决这种问题呢?个人以为方式一可能是目前性价比最好的解决方式吧。

一点体会

         JBPM看来在国内并没有得到很多的应用,资料都比较稀缺,唯一的参考手册也是非常的浅显和简单。而其源码的注释不是很好,这在国外的开源软件中是很罕见的。

         JBPM提供的某些API不是很全,举个例子,假定我要查询某个角色某个时间段的所有任务列表。因为TaskMgmtSession仅提供了findXXXTaskInstances(String actorId)方法,所以我只能先查询出该角色的所有任务列表,采用如下代码:

java 代码
  1. TaskMgmtSession taskMgmtSession = context.getTaskMgmtSession();   
  2.  List tasks = taskMgmtSession.findPooledTaskInstances(actorId);  

        然后再在内存中使用类似如下代码进行过滤。

java 代码
  1. if (task.getName().equals(taskName) && !task.hasEnded())  

这种方式太笨拙和低效了。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值