1. 程式人生 > >工作流引擎Activiti使用總結

工作流引擎Activiti使用總結

1.簡單介工作流引擎與Activiti

對於工作流引擎的解釋請參考百度百科:工作流引擎

1.1 我與工作流引擎

在第一家公司工作的時候主要任務就是開發OA系統,當然基本都是有工作流的支援,不過當時使用的工作流引擎是公司一些牛人開發的(據說是用一個開源的引擎修改的),名稱叫CoreFlow;功能相對Activiti來說比較弱,但是能滿足日常的使用,當然也有不少的問題所以後來我們只能修改引擎的程式碼打補丁。

現在是我工作的第二家公司,因為要開發ERP、OA等系統需要使用工作流,在專案調研階段我先搜尋資料選擇使用哪個開源工作流引擎,最終確定了Activiti5並基於公司的架構做了一些DEMO。

1.2 Activiti與JBPM5?

對於Activiti、jBPM4、jBPM5我們應該如何選擇,在InfoQ上有一篇文章寫的很好,從大的層面比較各個引擎之間的差異,請參考文章:縱觀jBPM:從jBPM3到jBPM5以及Activiti5

1.3 Activiti資料

2.初次使用遇到問題收集

因為Activiti剛剛退出不久所以資料比較空缺,中文資料更是少的可憐,所以開始的時候一頭霧水(雖然之前用過工作流,但是感覺差距很多),而且官方的手冊還不是很全面;所以我把我在學習使用的過程遇到的一些疑問都羅列出來分享給大家;以下幾點是我遇到和想到的,如果你還有什麼疑問可以在評論中和我交流再補充。

2.1 部署流程圖後中文亂碼

亂碼是一直纏繞著國人的問題,之前各個技術、工具出現亂碼的問題寫過很多文章,這裡也不例外……,Activiti的亂碼問題在流程圖中。

流程圖的亂碼如下圖所示:

通過*.bpmn20.xml部署後中文出現亂碼

解決辦法有兩種:

2.1.1 修改原始碼方式

修改原始碼

org.activiti.engine.impl.bpmn.diagram.ProcessDiagramCanvas

在構造方法

public ProcessDiagramCanvas(int width, int height)
中有一行程式碼是設定字型的,預設是用Arial字型,這就是亂碼產生的原因,把字改為本地的中文字型即可,例如:
?
1 Font font = new Font("WenQuanYi Micro Hei", Font.BOLD, 11);

當然如果你有配置檔案讀取工具那麼可以設定在*.properties檔案中,我就是這麼做的:

?
1 Font font = new Font(PropertyFileUtil.get("activiti.diagram.canvas.font"), Font.BOLD, 11);

5.12版本開始支援設定字型名稱,在引擎中新增如下設定,在生成圖片時即可使用微軟雅黑設定圖片中的文字。

?
1 <property name="activityFontName" value="微軟雅黑"></property>

2.1.2 使用壓縮包方式部署

Activiti支援部署*.bpmn20.xml、bar、zip格式的流程定義。

使用Activit Deisigner工具設計流程圖的時候會有三個型別的檔案:

  • .activiti設計工具使用的檔案

  • .bpmn20.xml設計工具自動根據.activiti檔案生成的xml檔案

  • .png流程圖圖片

解決辦法就是把xml檔案和圖片檔案同時部署,因為在單獨部署xml檔案的時候Activiti會自動生成一張流程圖的圖片檔案,但是這樣在使用的時候座標和圖片對應不起來……

所以把xml和圖片同時部署的時候Activiti自動關聯xml和圖片,當需要獲取圖片的時候直接返回部署時壓縮包裡面的圖片檔案,而不是Activiti自動生成的圖片檔案

2.1.2.1 使用工具打包Bar檔案

在“Package Explorer”檢視中右鍵專案名稱然後點選“Create deployment artifacts”,會在src目錄中建立deployment資料夾,裡面包含*.bar檔案.

2.1.2.2 使用Ant指令碼打包Zip檔案

這也是我們採用的辦法,你可以手動選擇xml和png打包成zip格式的檔案,也可以像我們一樣採用ant target的方式打包這兩個檔案。

123456789101112
<?xml version="1.0" encoding="UTF-8"?><project name="foo">    <property name="workflow.definition" value="foo-common-core/src/main/resources/diagrams" />    <property name="workflow.deployments" value="foo-common-core/src/main/resources/deployments" /><target name="workflow.package.oa.leave">		<echo>打包流程定義及流程圖::OA-請假</echo>		<zip destfile="${workflow.deployments}/oa/leave.zip" basedir="${workflow.definition}/oa/leave" update="true"			includes="*.xml,*.png" />	</target></project>

這樣當修改流程定義檔案後只要執行ant命令就可以打包了:

ant workflow.package.oa.leave

現在部署bar或者zip檔案檢視流程圖圖片就不是亂碼了,而是你的壓縮包裡面的png檔案。

2.2 使用引擎提供的Form還是自定義業務Form

2.2.1 引擎提供的Form

定義表單的方式在每個Task標籤中定義extensionElementsactiviti:formProperty即可,到達這個節點的時候可以通過API讀取表單元素。

Activiti官方的例子使用的就是在流程定義中設定每一個節點顯示什麼樣的表單哪些欄位需要顯示、哪些欄位只讀、哪些欄位必填。

但是這種方式僅僅適用於比較簡單的流程,對於稍微複雜或者頁面需要業務邏輯的判斷的情況就不適用了。

對於資料的儲存都是在引擎的表中,不利於和其他表的關聯、對整個系統的規劃也不利!

2.2.2 自定義業務Form

這種方式應該是大家用的最多的了,因為一般的業務系統業務邏輯都會比較複雜,而且資料庫中很多表都會有依賴關係,表單中有很多狀態判斷。

例如我們的系統適用jQuery UI作為UI,有很多javascript程式碼,頁面的很多操作需要特殊處理(例如:多個選項的互斥、每個節點根據型別和操作人顯示不同的按鈕);基本每個公司都有一套自己的UI風格,要保持多個系統的操作習慣一致只能使用自定義表單才能滿足。

2.3 業務和流程的關聯方式

這個問題在群裡面很多人都問過,這也是我剛剛開始迷惑的地方;

後來看了以下API發現RuntimeService有兩個方法:

2.3.1 startProcessInstanceByKey

javadoc對其說明:

startProcessInstanceByKey(String processDefinitionKey, Map variables) 
          Starts a new process instance in the latest version of the process definition with the given key

其中businessKey就是業務ID,例如要申請請假,那麼先填寫登記資訊,然後(儲存+啟動流程),因為請假是單獨設計的資料表,所以儲存後得到實體ID就可以把它傳給processInstanceBusinessKey方法啟動流程。當需要根據businessKey查詢流程的時候就可以通過API查詢:

?
1 runtimeService.createProcessInstanceQuery().processInstanceBusinessKey(processInstanceBusinessKey, processDefinitionKey)

建議資料庫冗餘設計:在業務表設計的時候新增一列:PROCESS_INSTANCE_ID varchar2(64),在流程啟動之後把流程ID更新到業務表中,這樣不管從業務還是流程都可以查詢到對方!

特別說明: 此方法啟動時自動選擇最新版本的流程定義。

2.3.2 startProcessInstanceById

javadoc對其說明:

startProcessInstanceById(String processDefinitionId, String businessKey, Map variables) 
          Starts a new process instance in the exactly specified version of the process definition with the given id.

processDefinitionId:這個引數的值可以通過repositoryService.createProcessDefinitionQuery()方法查詢,對應資料庫:ACT_RE_PROCDEF;每次部署一次流程定義就會新增一條資料,同名的版本號累加。

特別說明: 此可以指定不同版本的流程定義,讓使用者多一層選擇。

2.3.3 如何選擇

建議使用startProcessInstanceByKey,特殊情況需要使用以往的版本選擇使用startProcessInstanceById

2.4 同步使用者資料

這個問題也是比較多的人詢問過,Activiti支援對任務分配到:指定人、指定組、兩者組合,而這些人和組的資訊都儲存在ACT_ID..表中,有自己的使用者和組(角色)管理讓很多人不知所措了;原因是因為每個系統都會存在一個許可權管理模組(維護:使用者、部門、角色、授權),不知道該怎麼和Activiti同步。

2.4.1 建議處理方式

Activiti有一個IdentityService介面,通過這個介面可以操控Activiti的ACT_ID_*表的資料,一般的做法是用業務系統的許可權管理模組維護使用者資料,當進行CRUD操作的時候在原有業務邏輯後面新增同步到Activiti的程式碼;例如新增一個使用者時同步Activiti User的程式碼片段:

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849