5 安裝 jBPM 的 Eclipse 開(kāi)發(fā)插件
有個(gè)輔助工具開(kāi)發(fā)起來(lái)方便一點(diǎn),只不過(guò)現(xiàn)在 jBPM 的開(kāi)發(fā)工具插件功能還不算太強(qiáng),也就一個(gè)“項(xiàng)目創(chuàng)建向?qū)?#8221;的功能,讓你:
(1)不用再去配置 classpath 庫(kù)的引用了
(2)直接得到了一個(gè) jBPM 的項(xiàng)目初始結(jié)構(gòu)
其實(shí)吧,開(kāi)發(fā) jBPM 也不需要什么插件工具,在熟練了以后,庫(kù)引用了項(xiàng)目初始結(jié)構(gòu)都可以手工創(chuàng)建。
插件不用再去下載了, jbpm-starters-kit-3.1.1 包里就有,目錄地址如下: D:\jbpm-starters-kit-3.1.1\jbpm-designer\jbpm-gpd-feature\eclipse ,插件的安裝方式是鏈接式還是直接復(fù)制式,任選吧。不懂的就去看看《 Eclipse 從入門(mén)精通》這本書(shū),在前面章節(jié)都有講到。另外,注明一下 Eclipse 的版本我是用 3.2 ,插件和 Eclispe 版本相關(guān)的,要注意了。
如果安裝成功,則 Eclipse 首選項(xiàng)里多了一個(gè) JBoss jBPM ,另外我們也需要到這個(gè) jBPM 的首選項(xiàng)里做一些配置工作――指定 jBPM 的安裝路徑(如下圖所示)。這個(gè)配置主要是為了找到 jbpm 下的各種 jar 包,好讓 Eclipse 設(shè)置項(xiàng)目的庫(kù)引用。本文指向路徑是 d:\jbpm-starters-kit-3.1.1\jbpm.3
6 jBPM 的 Hello World
6.1 新建jBPM項(xiàng)目
主菜單“文件->新建->項(xiàng)目”,在彈出的對(duì)話框里,有“ Process Project ”項(xiàng),如下圖所示:
選上好,單擊“下一步”,起個(gè)名“ myjbpm ”,然后就可以單擊“完成”了。然后就生成了如下圖所示的一個(gè)項(xiàng)目結(jié)構(gòu):
這個(gè)項(xiàng)目和通常 Eclipse 的項(xiàng)目結(jié)構(gòu)有點(diǎn)不同,不過(guò)這是一個(gè)現(xiàn)在非常流行的項(xiàng)目結(jié)構(gòu), src/java 存放源文件, test/java 存放相應(yīng)的 JUnit 單元測(cè)試代碼。如果你用 Maven 來(lái)編譯構(gòu)建項(xiàng)目,對(duì)這種目錄結(jié)構(gòu)一定不陌生。
項(xiàng)目創(chuàng)建起了,介紹一下里面的文件吧:
l MessageActionHandler ,自動(dòng)生成的一個(gè) ActionHandler 。不想要可以刪掉。
l ehcache.xml cache 的配置文件,里面有很詳解的英文說(shuō)明。沒(méi)有必要可以不用改它。
l hibernate.cfg.xml jBPM 是用 Hibernate 進(jìn)行工作流的數(shù)據(jù)存儲(chǔ)的,這個(gè)就是 Hibernate 的配置文件。后面我們將講到如何配置這個(gè)文件。
l jbpm.cfg.xml jbpm 本身的配置文件?,F(xiàn)在是空的,它用的是缺省配置,你想知道有哪些配置就去看這個(gè)文件 D:\jbpm-starters-kit-3.1.1\jbpm.3\src\java.jbpm\org\jbpm\default.jbpm.cfg.xml
l log4j.properties 這個(gè)是日志 API 包 log4j 的配置文件,用過(guò) log4j 的都知道。
l SimpleProcessTest.java 這個(gè)是對(duì)最重要的流程配置文件的 processdefinition.xml 單元測(cè)試代碼。這里表?yè)P(yáng)一點(diǎn), jBPM 的優(yōu)良設(shè)計(jì)使得它的可測(cè)試性非常之高,喜歡寫(xiě) t 單元測(cè)試的人有福了。
l gpd.xml 用于生成流程圖的定義文件。都是一些方框的坐標(biāo)和長(zhǎng)寬
l processdefinition.xml 這個(gè)是對(duì)最重要的流程配置文件,以后寫(xiě)流程要經(jīng)常和它打交道。
l processimage.jpg 一個(gè)流程圖
從項(xiàng)目結(jié)構(gòu)來(lái)看,我們沒(méi)有看到 JSP 網(wǎng)頁(yè)程序,也沒(méi)有看到 GUI 客戶端程序,這些代碼都是要我們以后開(kāi)發(fā)中來(lái)寫(xiě)的。但本文不準(zhǔn)備用 JSP 、 GUI ( Swing 、 SWT )來(lái)做示例,而是用 JUnit 代碼來(lái)做使用 jBPM 客戶端來(lái)演示。因?yàn)?jBPM 實(shí)際上是一個(gè)后臺(tái)框架,至于前臺(tái)是 JSP 還是 Swing 還是無(wú)界面的 java.class 都是無(wú)關(guān)緊要的。在教程里用無(wú)界面的 java.class 來(lái)做客戶端則更方便一些,如果進(jìn)一步采用 JUnit ,則這樣的 java.class 同時(shí)還具備了單元測(cè)試的功能。以后就是用 JSP 寫(xiě)了 WEB 頁(yè)面,我們還是可以用這些 JUnit 程序來(lái)做單元測(cè)試,避免了頻繁的鼠標(biāo)點(diǎn)按 WEB 頁(yè)面這樣的力氣活。所以在 jBPM 自帶的英文教程里都是一個(gè) JUnit 程序,不仔佃看還真摸不著頭腦。
6.2 修改hibernate.cfg.xml
hibernate.cfg.xml 的默認(rèn)設(shè)置是用 HSQL ,這是一個(gè)內(nèi)存數(shù)據(jù)庫(kù),這種內(nèi)存數(shù)據(jù)庫(kù)用來(lái)代替項(xiàng)目實(shí)際所用的數(shù)據(jù)庫(kù)來(lái)做單元測(cè)試挺不錯(cuò)的。不過(guò)我們這里是要試試用 MySQL 、 Oracle ,那就改一下設(shè)置吧。
注:配置值可參考 D:\jbpm-starters-kit-3.1.1\jbpm-db 對(duì)應(yīng)子目錄下的 hibernate.properties 文件。
1 、 MySQL 的更改如下:
<property name="hibernate.dialect">org.hibernate.dialect.MySQLDialect</property>
<property name="hibernate.connection.driver_class">com.mysql.jdbc.Driver</property>
<property name="hibernate.connection.url">jdbc:mysql://localhost:3306/jbpm</property>
<property name="hibernate.connection.username">root</property>
<property name="hibernate.connection.password">123456</property>
2 、 Oracle 的更改如下:
<property name="hibernate.dialect">org.hibernate.dialect.OracleDialect</property>
<property name="hibernate.connection.driver_class">oracle.jdbc.driver.OracleDriver</property>
<property name="hibernate.connection.url">jdbc:oracle:thin:@192.168.123.10:1521:wxxrDB</property>
<property name="hibernate.connection.username">chengang</property>
<property name="hibernate.connection.password">chengang</property>
如果你裝了 Oracle 的客戶端,并且 D:\oracle\ora92\network\ADMIN\tnsnames.ora 里做了如下的設(shè)置
WXXRDB_192.168.123.10 =
(DESCRIPTION =
(ADDRESS_LIST =
(ADDRESS = (PROTOCOL = TCP)(HOST = 192.168.123.10)(PORT = 1521))
)
(CONNECT_DATA =
(SID = wxxrDB)
(SERVER = DEDICATED)
)
)
則 Oracle 的 hibernate.connection.url 項(xiàng)也可以設(shè)為: jdbc:oracle:oci:@WXXRDB_192.168.123.10
6.3 完善庫(kù)引用
雖然 jBPM 在創(chuàng)建項(xiàng)目之初給我們?cè)O(shè)置好了庫(kù)引用,如下圖
但后面運(yùn)行時(shí)還是報(bào)一些 NoClassDefFoundError 異常,如沒(méi)有對(duì) hibernate3.jar 的引用導(dǎo)致下面的錯(cuò)誤
java.lang.NoClassDefFoundError: org/hibernate/Session
at org.jbpm.persistence.db.DbPersistenceServiceFactory.openService(DbPersistenceServiceFactory.java:55)
at org.jbpm.svc.Services.getService(Services.java:136)
.......
所以我們要為本文的實(shí)例完善庫(kù)引用。主要是把 MySQL 和 Oracle 的 JDBC 庫(kù)、以及 Hibernate 的 hibernate3.jar 加入到項(xiàng)目的庫(kù)引用中。
(1) 找到缺少的 jar 包
l mysql 的 jdbc 包,在 D:\jbpm-starters-kit-3.1.1\jbpm-db\mysql\lib 目錄里
l oracle 的 jdbc 包, jbmp 中沒(méi)有包含(可能是沒(méi)拿到 oracle 授權(quán)),我們可以自已去 oracle 網(wǎng)站上下載,或者去 oracle 安裝目錄 D:\oracle\ora92\jdbc\lib 找 ojdbc14.jar (我們公司用的是 Oracle9i )
l Hibernate3.jar 在目錄 D:\jbpm-starters-kit-3.1.1\jbpm.3\lib\hibernate 里。
(2) 在項(xiàng)目里創(chuàng)建一個(gè) lib 目錄,將這三個(gè) jar 復(fù)制到 lib 目錄。
(3) 如下圖設(shè)置三 jar 包的庫(kù)引用
6.4 開(kāi)始HellorWorld
這里是一個(gè)很簡(jiǎn)單的請(qǐng)假流程,請(qǐng)假人提交假單給經(jīng)理審批,經(jīng)理審批后結(jié)束。要說(shuō)明的是,這個(gè)流程并不嚴(yán)謹(jǐn),比如經(jīng)理不通過(guò)流程應(yīng)該到哪?不過(guò)這并不防礙拿它來(lái)做示例,螃蟹還得一個(gè)一個(gè)的吃。我們先拿這一桿子捅到底的流程做一個(gè)最簡(jiǎn)單的示例,從整體上對(duì) jBPM 工作流開(kāi)發(fā)有概念先。然后我們?cè)俾S富。
1 、定義流程
流程的定義文件是 processdefinition.xml ,這個(gè)是一個(gè)關(guān)鍵文件, jBPM 的很大一部份內(nèi)容都是關(guān)于它的。在這里我們把原來(lái)自動(dòng)生成的內(nèi)容,稍做改動(dòng):
<?xml version="1.0" encoding="GBK"?>
<process-definition xmlns="urn:jbpm.org:jpdl-3.1" name="helloworld">
<!-- 申請(qǐng) -->
<start-state name="request">
<task>
<controller>
<variable name="name" />
<variable name="day" />
<variable name="note" />
</controller>
</task>
<!-- 流程轉(zhuǎn)向 -->
<transition name="to_confirm" to="confirm">
<action name="requestAction"
class ="cn.com.chengang.jbpm.RequestAction">
<reason> 我要請(qǐng)假 </reason>
</action>
</transition>
</start-state>
<!-- 審批 -->
<state name="confirm">
<transition name="to_end" to="end">
<action name="finishAction"
class ="cn.com.chengang.jbpm.ConfirmAction" />
</transition>
</state>
<!-- 結(jié)束 -->
<end-state name="end" />
</process-definition>
說(shuō)明:
流程的名稱(chēng)改成了 helloworld 。(呵呵,也就是這里和 helloworld 有關(guān)了)
<controller> 標(biāo)簽定義了三個(gè)數(shù)據(jù):姓名、請(qǐng)假天數(shù)、說(shuō)明。
<transition> 標(biāo)簽定了 request 節(jié)點(diǎn)的一個(gè)流程轉(zhuǎn)向,這里是轉(zhuǎn)到 confirm 節(jié)點(diǎn)。
<action> 標(biāo)簽定義了流程由一個(gè)節(jié)點(diǎn)轉(zhuǎn)到另一個(gè)節(jié)點(diǎn)時(shí),所要執(zhí)行的動(dòng)作,動(dòng)作封裝在一個(gè) ActionHandler 類(lèi)中。比如這里當(dāng) request 到 confirm 結(jié)點(diǎn)時(shí)將執(zhí)行 RequestAction 類(lèi)的 execute 方法。
FinishAction 下面還有一個(gè) <reason> (請(qǐng)假理由),它對(duì)應(yīng)于 FinshAction 的屬性 String reason 。
2 、 編寫(xiě) ActionHandler
在上面 processdefinition.xml 里我們定義了兩個(gè) ActionHandler : RequestAction 、 ConfirmAction 。其代碼如下:
package cn.com.chengang.jbpm;
import org.jbpm.graph.def.ActionHandler;
import org.jbpm.graph.exe.ExecutionContext;
public class RequestAction implements ActionHandler {
private static final long serialVersionUID = 1L;
private String reason;
public String getReason() {
return reason;
}
public void setReason(String reason) {
this .reason = reason;
}
public void execute(ExecutionContext context) throws Exception {
context.getContextInstance().setVariable("note", reason);
}
}
說(shuō)明: ExecutionContext 是一個(gè)貫通流程的容器。它是個(gè)大寶箱,里面啥玩意都有,后面將更深入的提到。這里的 reasion 就是 processdefinition.xml 中的 ” 我要請(qǐng)假 ”
package cn.com.chengang.jbpm;
import org.jbpm.graph.def.ActionHandler;
import org.jbpm.graph.exe.ExecutionContext;
public class ConfirmAction implements ActionHandler {
private static final long serialVersionUID = 1L;
public void execute(ExecutionContext context) throws Exception {
context.getContextInstance().setVariable("note", " 準(zhǔn)假 " );
}
}
OK ,后臺(tái)的程序就算寫(xiě)完了(前臺(tái)客戶端的程序還沒(méi)寫(xiě)),下面開(kāi)始部署。
6.5 部署processdefinition.xml
我們要把 processdefinition.xml 的流程定義的數(shù)據(jù)部署到數(shù)據(jù)庫(kù)中,因?yàn)?jBPM 在正式運(yùn)行的時(shí)候不是去讀 processdefinition.xml 文件,而是去讀數(shù)據(jù)庫(kù)中的流程定義。 這里寫(xiě)了一個(gè)個(gè) JUnit 程序來(lái)部署 processdefinition.xml ,當(dāng)然你用普通的 Java Main 也可以。
package com.sample;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.InputStream;
import junit.framework.TestCase;
import org.jbpm.JbpmConfiguration;
import org.jbpm.JbpmContext;
import org.jbpm.graph.def.ProcessDefinition;
/**
* 部署 processdefinition.xml
*
* @author chengang
*
*/
public class DeployProcessTest extends TestCase {
/**
* 在本方法執(zhí)行完畢后,檢查 jbpm_processdefinition 表會(huì)多了一條記錄
*
* @throws FileNotFoundException
*/
public void testDeployProcessDefinition() throws FileNotFoundException {
// 從 jbpm.cfg.xml 取得 jbpm 的配置
JbpmConfiguration config = JbpmConfiguration.getInstance();
// 創(chuàng)建一個(gè) jbpm 容器
JbpmContext jbpmContext = config.createJbpmContext();
// 由 processdefinition.xml 生成相對(duì)應(yīng)的流程定義類(lèi) ProcessDefinition
InputStream is = new FileInputStream("processes/simple/processdefinition.xml");
ProcessDefinition processDefinition = ProcessDefinition.parseXmlInputStream(is);
// 利用容器的方法將流程定義數(shù)據(jù)部署到數(shù)據(jù)庫(kù)上
jbpmContext.deployProcessDefinition(processDefinition);
// 關(guān)閉 jbpmContext
jbpmContext.close();
}
}
運(yùn)行此程序,在控制臺(tái)打印了一些日志,通過(guò)。如果出錯(cuò),仔佃閱讀出錯(cuò)信息以判斷錯(cuò)誤原因,并確定你按照前面兩節(jié):“修改 hibernate.cfg.xml ”和“完善庫(kù)引用”的內(nèi)容做好了設(shè)置。
6.6 從數(shù)據(jù)庫(kù)中的查看部署效果
無(wú)論是 MySQL 還是 Oracle ,查詢(xún) jbpm_processdefinition 表,你會(huì)發(fā)現(xiàn)多了一條記錄,如下圖 ( 以 PLSQL Developer 的顯示為例 )
依次檢查各表我們可以發(fā)現(xiàn)有如下變化:
并由此簡(jiǎn)單判斷出各表的作用,表中各字段的作用由字段名也能知曉一二。
jbpm_processdefinition
一個(gè)流程定義文件對(duì)應(yīng)一條記錄,可記錄多個(gè)流程定義文件,可記錄一個(gè)流程定義文件的對(duì)個(gè)版本。
jbpm_action
記錄 ActionHandler 的對(duì)象實(shí)例(以名稱(chēng)為標(biāo)識(shí))
jbpm_delegation
記錄了 ActionHandler 全類(lèi)名,以便于用反射方式來(lái)加載
jbpm_envent
它的 transition 引用了 Jbpm_transition 表的 id ,再看其它字段,估計(jì)此表是表示流程轉(zhuǎn)向事件的一個(gè)實(shí)例,或者是一個(gè)各表之間的聯(lián)接表。
jbpm_node
流程結(jié)點(diǎn)
jbpm_transition
流程的轉(zhuǎn)向定義
jbpm_variableaccess
流程中攜帶的變量。 ACCESS 字段是這些變量的讀寫(xiě)權(quán)限
作者簡(jiǎn)介
陳剛,廣西桂林人,著作有《Eclipse從入門(mén)到精通》
您可以通過(guò)其博客了解更多信息和文章:
http://www.ChenGang.com.cn