1. 程式人生 > >activiti工作流,駁回問題詳細解析(尤其會籤的駁回問題)

activiti工作流,駁回問題詳細解析(尤其會籤的駁回問題)

最近因為專案需要,所以研究了一下工作流,然而,我們都知道,國外的工作流框架,是不支援駁回問題的!這是其一!第二!專案的工作流要實現任意節點的流轉,這也是國外工作流框架所不具備的!因此,國外的工作流框架是沒有一個可以真正滿足我們的業務要求的,然而國內的工作流架構,我也沒有聽說過哪個可以支援中國式工作流!因此,在技術選型上面選擇了相對比較火的activiti框架!當時我們的原則就是儘可能的不改變框架,而實現我們的業務邏輯,因為,工作流這一大模組是本屌全權負責,就研究了一通!本專案實現了工作流節點的任意流轉,工作流回退(包含會籤),等等一系列操作,至於研究會籤問題的辛酸淚就不在多說,本節也不多說其他功能,主要說工作流回退,特別強調會籤回退,以及網上一些會籤回退的問題!
本屌在研究工作流的時候,曾參考csdn上某一位同志的部落格(點選藍色文字可以進入),但是在使用的時候,發現會籤是有一些問題的,所以對程式碼進行了一些調整,驗證完美實現會籤駁回
另外,在上程式碼之前一定宣告一點,大家一定注意我在完成任務的時候,進行了一系類的邏輯判斷,此判斷尤其重要,關係到會籤是否真正的沒有問題,曾經我出現過會籤駁回之後審批會出現多條記錄的情況,因此,希望大家一定要看仔細程式碼,最後:宣告一點,前面已經說了,本屌參考了一篇部落格,說明出處,因此有雷同程式碼,不要疑惑和糾結!廢話不在多少,直接上程式碼!
下面為核心程式碼,但是會籤的時候在審批要注意,在核心程式碼之後我會貼出我對會籤審批的處理
    /**
     * 
     * @param taskId 任務id
     * @param msg 批註
     * @param endActivityId 結束節點的activitiyId
     * @throws Exception
     */
    public void turnBackNew(String taskId, String msg, String endActivityId) throws Exception {
        Map<String, Object> variables;
        // 取得當前任務
        HistoricTaskInstance currTask = historyService
                .createHistoricTaskInstanceQuery().taskId(taskId)
                .singleResult();
        // 取得流程例項
ProcessInstance instance = runtimeService .createProcessInstanceQuery() .processInstanceId(currTask.getProcessInstanceId()) .singleResult(); if (instance == null) { throw new RuntimeException("流程已結束"); } variables = instance.getProcessVariables(); // 取得流程定義 ProcessDefinitionEntity definition = (ProcessDefinitionEntity) ((RepositoryServiceImpl) repositoryService) .getDeployedProcessDefinition(currTask .getProcessDefinitionId()); if (definition == null) { throw new RuntimeException("流程定義未找到"); } // 取得上一步活動 ActivityImpl currActivity = ((ProcessDefinitionImpl) definition) .findActivity(currTask.getTaskDefinitionKey()); List<ActivityImpl> rtnList = new ArrayList<>(); List<ActivityImpl> tempList = new ArrayList<>(); List<ActivityImpl> activities = iteratorBackActivity( taskId, currActivity, rtnList, tempList ); if (activities == null || activities.size() <= 0) throw new RuntimeException("沒有可以選擇的駁回節點!"); List<Task> list = taskService.createTaskQuery().processInstanceId(instance.getId()).list(); for (Task task : list) { if (!task.getId().equals(taskId)) { task.setAssignee("排除標記"); commitProcess(task.getId(), null, endActivityId); } } turnTransition(taskId, activities.get(0).getId(), null); } /** * @param taskId 當前任務ID * @param variables 流程變數 * @param activityId 流程轉向執行任務節點ID<br> * 此引數為空,預設為提交操作 * @throws Exception */ private void commitProcess(String taskId, Map<String, Object> variables, String activityId) throws Exception { if (variables == null) { variables = new HashMap<String, Object>(); } // 跳轉節點為空,預設提交操作 if (activityId == null || activityId.equals("")) { taskService.complete(taskId, variables); } else {// 流程轉向操作 turnTransition(taskId, activityId, variables); } } /** * 清空指定活動節點流向 * * @param activityImpl 活動節點 * @return 節點流向集合 */ private List<PvmTransition> clearTransition(ActivityImpl activityImpl) { // 儲存當前節點所有流向臨時變數 List<PvmTransition> oriPvmTransitionList = new ArrayList<PvmTransition>(); // 獲取當前節點所有流向,儲存到臨時變數,然後清空 List<PvmTransition> pvmTransitionList = activityImpl .getOutgoingTransitions(); for (PvmTransition pvmTransition : pvmTransitionList) { oriPvmTransitionList.add(pvmTransition); } pvmTransitionList.clear(); return oriPvmTransitionList; } /** * 還原指定活動節點流向 * * @param activityImpl 活動節點 * @param oriPvmTransitionList 原有節點流向集合 */ private void restoreTransition(ActivityImpl activityImpl, List<PvmTransition> oriPvmTransitionList) { // 清空現有流向 List<PvmTransition> pvmTransitionList = activityImpl .getOutgoingTransitions(); pvmTransitionList.clear(); // 還原以前流向 for (PvmTransition pvmTransition : oriPvmTransitionList) { pvmTransitionList.add(pvmTransition); } } /** * 流程轉向操作 * * @param taskId 當前任務ID * @param activityId 目標節點任務ID * @param variables 流程變數 * @throws Exception */ private void turnTransition(String taskId, String activityId, Map<String, Object> variables) throws Exception { // 當前節點 ActivityImpl currActivity = findActivitiImpl(taskId, null); // 清空當前流向 List<PvmTransition> oriPvmTransitionList = clearTransition(currActivity); // 建立新流向 TransitionImpl newTransition = currActivity.createOutgoingTransition(); // 目標節點 ActivityImpl pointActivity = findActivitiImpl(taskId, activityId); // 設定新流向的目標節點 newTransition.setDestination(pointActivity); // 執行轉向任務 taskService.complete(taskId, variables); // 刪除目標節點新流入 pointActivity.getIncomingTransitions().remove(newTransition); // 還原以前流向 restoreTransition(currActivity, oriPvmTransitionList); } /** * 迭代迴圈流程樹結構,查詢當前節點可駁回的任務節點 * * @param taskId 當前任務ID * @param currActivity 當前活動節點 * @param rtnList 儲存回退節點集合 * @param tempList 臨時儲存節點集合(儲存一次迭代過程中的同級userTask節點) * @return 回退節點集合 */ private List<ActivityImpl> iteratorBackActivity(String taskId, ActivityImpl currActivity, List<ActivityImpl> rtnList, List<ActivityImpl> tempList) throws Exception { // 查詢流程定義,生成流程樹結構 ProcessInstance processInstance = findProcessInstanceByTaskId(taskId); // 當前節點的流入來源 List<PvmTransition> incomingTransitions = currActivity .getIncomingTransitions(); // 條件分支節點集合,userTask節點遍歷完畢,迭代遍歷此集合,查詢條件分支對應的userTask節點 List<ActivityImpl> exclusiveGateways = new ArrayList<ActivityImpl>(); // 並行節點集合,userTask節點遍歷完畢,迭代遍歷此集合,查詢並行節點對應的userTask節點 List<ActivityImpl> parallelGateways = new ArrayList<ActivityImpl>(); // 遍歷當前節點所有流入路徑 for (PvmTransition pvmTransition : incomingTransitions) { TransitionImpl transitionImpl = (TransitionImpl) pvmTransition; ActivityImpl activityImpl = transitionImpl.getSource(); String type = (String) activityImpl.getProperty("type"); /** * 並行節點配置要求:<br> * 必須成對出現,且要求分別配置節點ID為:XXX_start(開始),XXX_end(結束) */ if ("parallelGateway".equals(type)) {// 並行路線 String gatewayId = activityImpl.getId(); String gatewayType = gatewayId.substring(gatewayId .lastIndexOf("_") + 1); if ("START".equals(gatewayType.toUpperCase())) {// 並行起點,停止遞迴 return rtnList; } else {// 並行終點,臨時儲存此節點,本次迴圈結束,迭代集合,查詢對應的userTask節點 parallelGateways.add(activityImpl); } } else if ("startEvent".equals(type)) {// 開始節點,停止遞迴 return rtnList; } else if ("userTask".equals(type)) {// 使用者任務 tempList.add(activityImpl); } else if ("exclusiveGateway".equals(type)) {// 分支路線,臨時儲存此節點,本次迴圈結束,迭代集合,查詢對應的userTask節點 currActivity = transitionImpl.getSource(); exclusiveGateways.add(currActivity); } } /** * 迭代條件分支集合,查詢對應的userTask節點 */ for (ActivityImpl activityImpl : exclusiveGateways) { iteratorBackActivity(taskId, activityImpl, rtnList, tempList); } /** * 迭代並行集合,查詢對應的userTask節點 */ for (ActivityImpl activityImpl : parallelGateways) { iteratorBackActivity(taskId, activityImpl, rtnList, tempList); } /** * 根據同級userTask集合,過濾最近發生的節點 */ currActivity = filterNewestActivity(processInstance, tempList); if (currActivity != null) { // 查詢當前節點的流向是否為並行終點,並獲取並行起點ID String id = findParallelGatewayId(currActivity); if (id == null || id.equals("")) {// 並行起點ID為空,此節點流向不是並行終點,符合駁回條件,儲存此節點 rtnList.add(currActivity); } else {// 根據並行起點ID查詢當前節點,然後迭代查詢其對應的userTask任務節點 currActivity = findActivitiImpl(taskId, id); } // 清空本次迭代臨時集合 tempList.clear(); // 執行下次迭代 iteratorBackActivity(taskId, currActivity, rtnList, tempList); } return rtnList; } /** * 根據當前節點,查詢輸出流向是否為並行終點,如果為並行終點,則拼裝對應的並行起點ID * * @param activityImpl 當前節點 * @return */ private String findParallelGatewayId(ActivityImpl activityImpl) { List<PvmTransition> incomingTransitions = activityImpl .getOutgoingTransitions(); for (PvmTransition pvmTransition : incomingTransitions) { TransitionImpl transitionImpl = (TransitionImpl) pvmTransition; activityImpl = transitionImpl.getDestination(); String type = (String) activityImpl.getProperty("type"); if ("parallelGateway".equals(type)) {// 並行路線 String gatewayId = activityImpl.getId(); String gatewayType = gatewayId.substring(gatewayId .lastIndexOf("_") + 1); if ("END".equals(gatewayType.toUpperCase())) { return gatewayId.substring(0, gatewayId.lastIndexOf("_")) + "_start"; } } } return null; } /** * 根據流入任務集合,查詢最近一次的流入任務節點 * * @param processInstance 流程例項 * @param tempList 流入任務集合 * @return */ private ActivityImpl filterNewestActivity(ProcessInstance processInstance, List<ActivityImpl> tempList) { while (tempList.size() > 0) { ActivityImpl activity_1 = tempList.get(0); HistoricActivityInstance activityInstance_1 = findHistoricUserTask( processInstance, activity_1.getId()); if (activityInstance_1 == null) { tempList.remove(activity_1); continue; } if (tempList.size() > 1) { ActivityImpl activity_2 = tempList.get(1); HistoricActivityInstance activityInstance_2 = findHistoricUserTask( processInstance, activity_2.getId()); if (activityInstance_2 == null) { tempList.remove(activity_2); continue; } if (activityInstance_1.getEndTime().before( activityInstance_2.getEndTime())) { tempList.remove(activity_1); } else { tempList.remove(activity_2); } } else { break; } } if (tempList.size() > 0) { return tempList.get(0); } return null; } /** * 查詢指定任務節點的最新記錄 * * @param processInstance 流程例項 * @param activityId * @return */ private HistoricActivityInstance findHistoricUserTask( ProcessInstance processInstance, String activityId) { HistoricActivityInstance rtnVal = null; // 查詢當前流程例項審批結束的歷史節點 List<HistoricActivityInstance> historicActivityInstances = historyService .createHistoricActivityInstanceQuery().activityType("userTask") .processInstanceId(processInstance.getId()).activityId( activityId).finished() .orderByHistoricActivityInstanceEndTime().desc().list(); if (historicActivityInstances.size() > 0) { rtnVal = historicActivityInstances.get(0); } return rtnVal; } /** * 根據任務ID和節點ID獲取活動節點 <br> * * @param taskId 任務ID * @param activityId 活動節點ID <br> * 如果為null或"",則預設查詢當前活動節點 <br> * 如果為"end",則查詢結束節點 <br> * @return * @throws Exception */ private ActivityImpl findActivitiImpl(String taskId, String activityId) throws Exception { // 取得流程定義 ProcessDefinitionEntity processDefinition = findProcessDefinitionEntityByTaskId(taskId); // 獲取當前活動節點ID if (activityId == null || "".equals(activityId)) { activityId = findTaskById(taskId).getTaskDefinitionKey(); } // 根據流程定義,獲取該流程例項的結束節點 if (activityId.toUpperCase().equals("END")) { for (ActivityImpl activityImpl : processDefinition.getActivities()) { List<PvmTransition> pvmTransitionList = activityImpl .getOutgoingTransitions(); if (pvmTransitionList.isEmpty()) { return activityImpl; } } } // 根據節點ID,獲取對應的活動節點 ActivityImpl activityImpl = ((ProcessDefinitionImpl) processDefinition) .findActivity(activityId); return activityImpl; } /** * 根據任務ID獲取流程定義 * * @param taskId 任務ID * @return * @throws Exception */ private ProcessDefinitionEntity findProcessDefinitionEntityByTaskId( String taskId) throws Exception { // 取得流程定義 ProcessDefinitionEntity processDefinition = (ProcessDefinitionEntity) ((RepositoryServiceImpl) repositoryService) .getDeployedProcessDefinition(findTaskById(taskId) .getProcessDefinitionId()); if (processDefinition == null) { throw new Exception("流程定義未找到!"); } return processDefinition; } /** * 根據任務ID獲得任務例項 * * @param taskId 任務ID * @return * @throws Exception */ private TaskEntity findTaskById(String taskId) throws Exception { TaskEntity task = (TaskEntity) taskService.createTaskQuery().taskId( taskId).singleResult(); if (task == null) { throw new Exception("任務例項未找到!"); } return task; } /** * 根據任務ID獲取對應的流程例項 * * @param taskId 任務ID * @return * @throws Exception */ private ProcessInstance findProcessInstanceByTaskId(String taskId) throws Exception { // 找到流程例項 ProcessInstance processInstance = runtimeService .createProcessInstanceQuery().processInstanceId( findTaskById(taskId).getProcessInstanceId()) .singleResult(); if (processInstance == null) { throw new Exception("流程例項未找到!"); } return processInstance; }
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39
  • 40
  • 41
  • 42
  • 43
  • 44
  • 45
  • 46
  • 47
  • 48
  • 49
  • 50
  • 51
  • 52
  • 53
  • 54
  • 55
  • 56
  • 57
  • 58
  • 59
  • 60
  • 61
  • 62
  • 63
  • 64
  • 65
  • 66
  • 67
  • 68
  • 69
  • 70
  • 71
  • 72
  • 73
  • 74
  • 75
  • 76
  • 77
  • 78
  • 79
  • 80
  • 81
  • 82
  • 83
  • 84
  • 85
  • 86
  • 87
  • 88
  • 89
  • 90
  • 91
  • 92
  • 93
  • 94
  • 95
  • 96
  • 97
  • 98
  • 99
  • 100
  • 101
  • 102
  • 103
  • 104
  • 105
  • 106
  • 107
  • 108
  • 109
  • 110
  • 111
  • 112
  • 113
  • 114
  • 115
  • 116
  • 117
  • 118
  • 119
  • 120
  • 121
  • 122
  • 123
  • 124
  • 125
  • 126
  • 127
  • 128
  • 129
  • 130
  • 131
  • 132
  • 133
  • 134
  • 135
  • 136
  • 137
  • 138
  • 139
  • 140
  • 141
  • 142
  • 143
  • 144
  • 145
  • 146
  • 147
  • 148
  • 149
  • 150
  • 151
  • 152
  • 153
  • 154
  • 155
  • 156
  • 157
  • 158
  • 159
  • 160
  • 161
  • 162
  • 163
  • 164
  • 165
  • 166
  • 167
  • 168
  • 169
  • 170
  • 171
  • 172
  • 173
  • 174
  • 175
  • 176
  • 177
  • 178
  • 179
  • 180
  • 181
  • 182
  • 183
  • 184
  • 185
  • 186
  • 187
  • 188
  • 189
  • 190
  • 191
  • 192
  • 193
  • 194
  • 195
  • 196
  • 197
  • 198
  • 199
  • 200
  • 201
  • 202
  • 203
  • 204
  • 205
  • 206
  • 207
  • 208
  • 209
  • 210
  • 211
  • 212
  • 213
  • 214
  • 215
  • 216
  • 217
  • 218
  • 219
  • 220
  • 221
  • 222
  • 223
  • 224
  • 225
  • 226
  • 227
  • 228
  • 229
  • 230
  • 231
  • 232
  • 233
  • 234
  • 235
  • 236
  • 237
  • 238
  • 239
  • 240
  • 241
  • 242
  • 243
  • 244
  • 245
  • 246
  • 247
  • 248
  • 249
  • 250
  • 251
  • 252
  • 253
  • 254
  • 255
  • 256
  • 257
  • 258
  • 259
  • 260
  • 261
  • 262
  • 263
  • 264
  • 265
  • 266
  • 267
  • 268
  • 269
  • 270
  • 271
  • 272
  • 273
  • 274
  • 275
  • 276
  • 277
  • 278
  • 279
  • 280
  • 281
  • 282
  • 283
  • 284
  • 285
  • 286
  • 287
  • 288
  • 289
  • 290
  • 291
  • 292
  • 293
  • 294
  • 295
  • 296
  • 297
  • 298
  • 299
  • 300
  • 301
  • 302
  • 303
  • 304
  • 305
  • 306
  • 307
  • 308
  • 309
  • 310
  • 311
  • 312
  • 313
  • 314
  • 315
  • 316
  • 317
  • 318
  • 319
  • 320
  • 321
  • 322
  • 323
  • 324
  • 325
  • 326
  • 327
  • 328
  • 329
  • 330
  • 331
  • 332
  • 333
  • 334
  • 335
  • 336
  • 337
  • 338
  • 339
  • 340
  • 341
  • 342
  • 343
  • 344
  • 345
  • 346
  • 347
  • 348
  • 349
  • 350
  • 351
  • 352
  • 353
  • 354
  • 355
  • 356
  • 357
  • 358
  • 359
  • 360
  • 361
  • 362
  • 363
  • 364
  • 365
  • 366
  • 367
  • 368
  • 369
  • 370
  • 371
  • 372
  • 373
  • 374
  • 375
  • 376
  • 377
  • 378
  • 379
  • 380
  • 381
  • 382
  • 383
  • 384
  • 385
  • 386
  • 387
  • 388
  • 389
  • 390
  • 391
  • 392
  • 393
  • 394
  • 395
  • 396
  • 397
  • 398
  • 399
  • 400
  • 401
  • 402
  • 403
  • 404
  • 405
  • 406
  • 407
  • 408
  • 409
  • 410
  • 411
  • 412
下面尤其重要的說明一點,會簽在提交的時候,一定要如我之判定,否則,會在多次駁回的時候,出現多條記錄,流程混亂的問題
public void submitTask(ProjectWorkflowRequest param, String userCode, String username, User user) throws Exception {
        String s = WorkflowReflectionLog.refectCompare(projectService.getProjectByProjectId(projectService.getNewestProjectIdByApplicationId(param.getProject().getTab1().getApplicationProjectId())), param.getProject());// 此段為本屌用反射記錄的審批人修改的內容的日誌(為我專案專屬所有,大家刪掉就好)
        //  if (true) throw new RuntimeException();
        projectService.applicationSubmit(param.getProject(), user);// 此段是為了滿足甲方需求,記錄的歷史,大家刪掉就好
        Task task = taskService.createTaskQuery().taskId(param.getTaskId()).singleResult();// 獲取當前任務
        String processInstanceId = task.getProcessInstanceId();
        taskService.addComment(param.getTaskId(), processInstanceId, param.getMsg());// 新增任務批註
        taskService.setVariableLocal(param.getTaskId(), "projectId", param.getProject().getTab1().getProjectId());// 此段也是為了記錄歷史而誕生,大家可以刪掉
        if (s != null) taskService.setVariableLocal(param.getTaskId(), "log", s);// 大家可以刪掉
        runtimeService.setVariables(task.getExecutionId(), new HashMap() {{
            this.put("LAST_USER", userCode);
            this.put("LAST_USER_NAME", username);
        }});// 大家可以刪掉

        runtimeService.setVariable(processInstanceId, "LAST_USER", username);// 大家可以刪掉
// 因為前臺標記駁回還是通過還是終止流程,分別用0,1,2標識,所以,你就當是三種情況,通過駁回,終止
        if ("1".equals(param.getFlag())) {// 通過
            switch (param.getType()) {
                case 1: {
                    String sid = task.getTaskDefinitionKey();// 獲取當前任務的activitiyId
                    List<Task> list = taskService.createTaskQuery().processInstanceId(task.getProcessInstanceId()).list();// 獲取當前流程的所有未完成任務
                    if (list.size() == 1) {// 如果只剩一個任務,可能是普通任務,可能是會籤任務
                        if (
                                "sid-71E0BFA1-AD56-45D6-907E-D4B61F445D84".equals(sid) ||
                                        "sid-9795199D-77F0-4F34-A0D6-E388525FA8BE".equals(sid) ||
                                        "sid-6340E7E8-2BA1-4B2F-B7CF-19FE31412E58".equals(sid)) {// 如果是會籤,則提交到下一個審批節點(並行閘道器之後的第一個節點)
//工作流任意流轉,流轉到下一個審批節點,因為我的會籤裡面包含三個節點,所以經過activitiyId來判定是不是這三個節點,如果是,則任意流轉節點,如果不是,則正常完成任務,注意:工作流程圖確定之後,每個節點的activityId是永恆不變的!因此,才可以做此判定
                            commitProcess(task.getId(), null, "sid-C6334E47-F8A8-430C-9A35-FA98CFD0C405");
                        } else if (
                                "sid-62D73895-5AB3-4818-8795-D8CCCDB59CE6".equals(sid) ||
                                        "sid-CFDED207-B694-478A-AE94-3DFFD486E77F".equals(sid) ||
                                        "sid-7A523F96-9F43-4C2C-979F-06B7BC9656B6".equals(sid)
                                ) else taskService.complete(task.getId());// 如果不是會籤的3個節點則直接審批
                    } else {
                        commitProcess(task.getId(), null, "sid-404150FC-8E2D-4582-9CFB-BC9FF43F09E1");
                    }
                }
                break;
                case 2: {
                    Map<String, Object> variables = new HashMap<>();
                    variables.put("a", param.getU1());
                    variables.put("b", param.getU2());
                    taskService.complete(param.getTaskId(), variables);
                }
                break;
                case 3: {
                    Map<String, Object> variables = new HashMap<>();
                    variables.put("flag", param.getFlag1());
                    taskService.complete(param.getTaskId(), variables);
                }
                break;
            }

        } else if ("0".equals(param.getFlag())) {// 拒絕 // // TODO: 2017/11/8 0008
            turnBackNew(param.getTaskId(), param.getMsg(), "sid-404150FC-8E2D-4582-9CFB-BC9FF43F09E1");
        } else if ("2".equals(param.getFlag())) {// 終止流程
            endProcess(param.getTaskId(), "sid-404150FC-8E2D-4582-9CFB-BC9FF43F09E1");
        }
    }
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39
  • 40
  • 41
  • 42
  • 43
  • 44
  • 45
  • 46
  • 47
  • 48
  • 49
  • 50
  • 51
  • 52
  • 53
  • 54
  • 55
  • 56
  • 57
  • 58
  • 59

相關推薦

activiti工作,駁回問題詳細解析(尤其駁回問題)

最近因為專案需要,所以研究了一下工作流,然而,我們都知道,國外的工作流框架,是不支援駁回問題的!這是其一!第二!專案的工作流要實現任意節點的流轉,這也是國外工作流框架所不具備的!因此,國外的工作流框架是沒有一個可以真正滿足我們的業務要求的,然而國內的工作流架構,我也沒有聽說過哪個可以支援中國式工作流!因此,在

簡易OA漫談之工作設計(四,怎麼支援)

會籤:就是一個步驟要所有人都審批完成才能完成。 上一步提交到會籤步驟時,該步驟的每個人都能收到一條待辦任務, 會籤審批人每次提交時都檢查該步驟所有會籤人是否已經審批,如果全部審批就產生下一手任務。 演示:http://wftest.bomye.cn/ 1、先建一個使用者組 2、給使用者組加兩個成

activiti工作,資料庫表解析

-- 釋出的流程資訊 存有id和名稱。   該id分別在“部署的流程” 和“通用資料” 中記錄 以該表為紐帶,關聯 “部署的流程act_re_procdef” 與“部署流程的配置資訊(該配置資訊記錄在‘通用資料’act_ge_bytearray表中)” select *

帶你瞭解什麼是Activiti工作Activiti工作資料庫表詳細介紹(23張表)

帶你五分鐘瞭解工作流 什麼是工作流 說到工作流,一圖勝萬言。 工作流 Georgakopoulos給出的工作流定義是:工作流是將一組任務組織起來以完成某個經營過程:定義了任務的觸發順序和觸發條件,每個任務可以由一個或多個軟體系統完成,也可以由一個或一組人完成,還可以由一個或多個

Activiti工作引擎添新丁:Flowable6.0(聽說無縫連線你換嗎?)

如果你在還糾結該選擇JPMB還是Acitiviti的時候,或者還在糾結於是否該從JPMB遷移到Activiti的陣營中的時候,很不幸地告訴你,Flowable6.0已經發布了。   是不是變得更糾結啦?!又多了一種選擇。   1、 什麼是Flowable?   

Activiti工作REST方式訪問URI的解析

在Activiti工作流中提供一種訪問方式為REST方式訪問。針對查詢字串的格式採用一個如下一個工具類解析Uri中的提供的各種資訊。 package org.activiti.explorer.util; import java.io.UnsupportedEncodin

Activiti工作引擎的官方例項配置詳細步驟

概述 學習Activiti工作流引擎,第一步自然是是下載官方提供的Demo原始碼來參考。 Activiti官網:activiti-5.22.0.zip 下載下來的原始碼結構為: war包裡就是官方示例 建立Java web專案

activiti工作資料庫表詳細介紹

Activiti的後臺是有資料庫的支援,所有的表都以ACT_開頭。 第二部分是表示表的用途的兩個字母標識。 用途也和服務的API對應。 ACT_RE_*: 'RE'表示repository。 這個字首的表包含了流程定義和流程靜態資源 (圖片,規則,等等)。 A

Activiti工作框架學習(一)之通用資料表詳細介紹

文/朱季謙 Activiti工作流引擎自帶了一套資料庫表,這裡面有一個需要注意的地方: 低於5.6.4的MySQL版本不支援時間戳或毫秒級的日期。更糟糕的是,某些版本在嘗試建立此類列時將引發異常,而其他版本則不會。執行自動建立/升級時,引擎將在執行DDL時更改它。使用DDL檔案方法時,既可以使用常規版本也可以

activiti工作系列-activitirume

例如 參與者 span fin 動態 完全 def ini gin 1、什麽是工作流? 工作流就是讓多個參與者之間按照某種預定義的規則傳遞文檔、信息或任務的過程,工作流由實體(Entity)、參與者(Participant)、流程定義(Flow Definition)、工作

Activiti工作引擎數據庫表結構

activit let repr 通過 運行時 工作流 用例 pro ext Activiti工作流引擎數據庫表結構 一、數據庫表的命名   Acitiviti數據庫中表的命名都是以ACT_開頭的。第二部分是一個兩個字符用例表的標識。此用例大體與服務API是匹配的。 l

Java Activiti 工作引擎 springmvc SSM 流程審批 後臺框架源碼

Activiti6.0工作流模塊----------------------------------------------------------------------------------------------------------1.模型管理 :web在線流程設計器、預覽流程xml、導出x

activiti工作EL表達式出現亂碼導致comment亂碼

com 重啟 down true 情況 工作流 原因 表達式 AC EL表達式出現亂碼的原因有很多,我記錄一下自己的情況。 在response和頁面都已設置utf-8編碼時仍然出現亂碼。 我修改了tomcat conf目錄下的server.xml文件 在原始配置上添加了u

Activiti6.0工作引擎深度解析與實戰

word boot 事件處理 數據庫表 系統 講解 數據表設計 helloword posit 第1章 課程介紹本課程將系統且深入源碼講解Activiti6.0工作流引擎的使用、配置、核心api以及BPMN2.0規範、數據庫設計及模型映射,Spring Boot2.0集成,

Java Activiti 工作引擎 流程審批 後臺框架源碼 springmvc SSM

物流查詢接口 主表 快速 font sql www. 物流 sock hit 博文來源:http://www.fhadmin.org/webnewsdetail1.html工作流模塊----------------------------------------------

工作3-activiti工作流程定義的CURD

activiti的流程定義API Explain:activiti對流程的增刪該查 (學習源自傳智播客itcast) 獲取引擎 ProcessEngine processEngine = ProcessEngine.getDefaultProcessEngine();

Activiti工作引擎學習總結

經過這三天對與Activiti的軟磨硬泡,也算是小有所獲。 想要學會工作流引擎其實很簡單,搞清楚processengine (Activiti引擎),並且理解其中幾大服務介面: 服務:    reposit

Activiti工作引擎學習及流程設計器的安裝(Eclipse外掛)

工作流簡介     我們先來描述一個在工作中經常用到的一個流程:請假     員工張三請假一天 主管王五批准     員工李四請假一天 主管王五不批准  &nbs

IDEA建立Activiti工作開發第一篇(Maven專案)

首先建立一個新的Maven專案,建立完成後我們要安裝Activiti的外掛,首先開啟FIle的setting功能,搜尋Plugins: 輸入actiBPM,然後點選搜尋: 如圖點選安裝,它就會下載安裝 安裝好後點擊應用: 然後重啟IDEA,點選File檔案中的

Activiti工作的環境配置

分享一下我老師大神的人工智慧教程吧。零基礎,通俗易懂!風趣幽默!http://www.captainbed.net/ 也歡迎大家轉載本篇文章。分享知識,造福人民,實現我們中華民族偉大復興!