前言

课设基于SSM框架,数据库采用mysql,主要业务交给activiti,版本控制利用github。

参考资料:

activiti学习小记

基于0中Activiti就是这么简单,做些笔记。

当前环境:按照Intellij 部署SSM框架中部署完成SSM框架,能tomcat启动,连接上数据库。

快速入门

在test\java\ml\shylocks\下新建一个class做activiti测试用。



pom.xml中加入以下代码,以便maven从中央仓库push activiti所需要的包

 <dependency>
<groupId>org.activiti</groupId>
<artifactId>activiti-engine</artifactId>
<version>5.22.0</version>
</dependency> <dependency>
<groupId>org.activiti</groupId>
<artifactId>activiti-spring</artifactId>
<version>5.22.0</version>
</dependency>
<dependency>
<groupId>org.codehaus.groovy</groupId>
<artifactId>groovy-all</artifactId>
<version>2.4.3</version>
</dependency>

1.4.1activiti.cfg.xml放入resource文件夹下,同时创建工作流也在同文件夹下:

测试1.4.3deploy()

在函数开头加入ProcessEngine processEngine = ProcessEngines.getDefaultProcessEngine();

所需要import的包为:



点击函数右侧



即可快速运行。

运行成功结果:



可能出现的错误为.bpmn文件错误命名、放置。

查询数据库,没有查到任何结果…?



本地mysql连接有问题,所以用的是云服务器上的mysql数据库,首先检测jdbc连接没有问题,再检测activit.cfg.xml,发现数据库字段不是自定义的库,而是复制的activitiDB。

再次查询activitiDB库,得到结果:



查询为何出错的同时,查找的helpful博文:烟火_:activiti入门

将acitivit用到的事务放在before里:

测试1.4.4startProcess()1.4.5queryTask()

导入processinstance包:



注意将processDefiKey改为创建的工作流id,不然会出现以下错误:



要是在deploy()执行后执行startProcess(),注意工作流id的一致,后续更改是无效的。

执行queryTask(),查询数据库,表项已添加:

测试1.4.6compileTask()

注意taskId是刚刚queryTask()执行的id,直接复制代码会出现异常。

执行后数据库表项:

小结

经过以上学习,activiti的大致流程如OA一样,由申请人发起事务,审批人审批事务,最终结束。

所有数据库端操作均由底层mybatis执行,只需要调用接口,可以大大简化操作流程。

延伸:Activiti工作流引擎主要表结构

概要设计

  • 设计草图

    系统总功能图:



    模块图:



    流程图:



    对于以上,可以抽象出以下业务:

Contorller类



对于校园用户体系来说,高权限用户对低权限用户具有管理权限,但由于activiti的局限性,无法建立班级——班主任,系(专业)——辅导员这样的一对一关系,想要做出完整的校园请假系统应建立对应的数据表,数据操作是利用联合查询。

班主任用户组与辅导员用户组,区别在于请假业务中学生请假时长不同,分配到的用户不同,权限实际上是相同的。

由图中可以看出,不同权限组可以访问的页面存在高度的耦合性,所以整合为以下页面:

Service类

为了将界面层与业务逻辑层分离,将所有对数据的操作包装在service类中

事务建模

请假事务:



节假日去向/寒暑假申请:

详细设计

基础部分略去,只列出实现过程中遇到的各种问题。

Model

  • 在使用mybatis封装的mapper对数据库进行操作时,创建的mapper总是提示NullPointerError,由于mapper是在applicationContext.xml中自动扫描的,手动加载没有问题,谷歌无数后才解决:

    • 在*mapper.java中,interface前加入@Repository:

    • 在*Controller.java创建mapper前加入@Autowired:

  • 由于activiti的数据表设计,用户的用户组管理(act_id_membership)与用户基本信息(act_id_user)不在一张表中,对于用户管理来说很不方便,需要将act_id_membership、act_id_user、act_id_group三表联合查询。mybatis对于数据库的操作封装在*Mapper.xml中,利用mybatis-generator生成三张表的相应mapper,在Actidusermapper.xml的resultmap标签中加入

<association property="group" javaType="ml.shylocks.entity.ActIdGroup" >
<id column="ID_" property="id" jdbcType="VARCHAR" />
<result column="REV_" property="rev" jdbcType="INTEGER" />
<result column="NAME_" property="name" jdbcType="VARCHAR" />
<result column="TYPE_" property="type" jdbcType="VARCHAR" />
</association>
<collection property="membership" javaType="ml.shylocks.entity.ActIdMembershipKey">
<id column="USER_ID_" property="userId" jdbcType="VARCHAR" />
<id column="GROUP_ID_" property="groupId" jdbcType="VARCHAR" />
</collection>

在mapper标签中加入

<select id="getUser" resultMap="BaseResultMap">
select *
from act_id_user A
join act_id_membership B on A.id_=B.user_id_
join act_id_group C on B.group_id_=C.id_
</select>
<select id="getUserByGroup" resultMap="BaseResultMap" parameterType="java.lang.String">
select *
from act_id_user A
join act_id_membership B on A.id_=B.user_id_
join act_id_group C on B.group_id_=C.id_
where B.group_id_=#{groupid,jdbcType=VARCHAR}
</select>

在actiduser.java中加入:

 public ActIdGroup group;
public ActIdMembershipKey membership;
public ActIdGroup getGroup() {
return group;
} public void setGroup(ActIdGroup group) {
this.group = group;
} public ActIdMembershipKey getMembership() {
return membership;
} public void setMembership(ActIdMembershipKey membership) {
this.membership = membership;
}

将actidusermapper.java改为:

@Repository
public interface UserMapper {
List<User> getUser();
List<User> getUserByGroup(String groupid);
}

View

  • 原本计划是前后端分离,利用vue.js写前端的,不过一个人完成整个项目时间不够,最后还是用jsp写的。在不使用iframe,framework的情况下,j2ee中想要实现类似django中的extends使用的是http://www.rapid-framework.org.cn/rapid,将父框架放入base.jsp,其他子页面利用rapid:block对父框架中相应内容进行重写。但是这种复写方式会导致利用Model写入父框架的java元素在子框架中无法显示,所以子页面利用$.load()加载。

    注意子页面中的js代码应当使用规范标签,加入type="text/javascript"。
  • html5中input新增了date的类型可以代替datetimepicker

Controller

  • 对于寒暑假登记、节假日去向事务来说,不应该将其声明为学生用户组,这样只会创建一个流程实例,应当在发布时对所有学生申明。

    • 前端部分使用select2插件选择学生:



      由于spring mvc 无法对多个同名 RequestParam 进行解析,在提交表单时将学生id数组以字符串形式传递:
if (data['candidate'] == "students") {
let c = $("#select2").val();
for (let x = 0; x < c.length; ++x) {
data['candidateList'] += '"' + c[x] + '",';
}
data['candidateList'] += "]";
}
- 后端接受时,利用jsonarray对其进行解析
JSONArray jsonArray = JSONArray.fromObject(form.get("candidateList"));
List<String> candidateList = new ArrayList<String>();
for (int i = 0; i < jsonArray.size(); ++i) candidateList.add(jsonArray.optString(i));
- 注意在`task.setAssignee()`后保存当前任务`taskservice.saveTask(task)`
  • 由于设计的业务流程中,学生填写/提出申请并未指定用户/用户组,所以当审批不通过时,应人工指定流程中当前任务的assignee,注意runtimeservice中只保存当前任务,找到第一个userTask的assignee应使用historyService:
Task nowTask = taskService.createTaskQuery().executionId(task.getExecutionId()).singleResult();
if (nowTask != null) {
HistoryService historyService = processEngine.getHistoryService();
HistoricTaskInstance historicTaskInstance = historyService.
createHistoricTaskInstanceQuery().
executionId(defId).list().get(0);
nowTask.setAssignee(historicTaskInstance.getAssignee());
nowTask.setCategory(historicTaskInstance.getCategory());
taskService.saveTask(nowTask);
}

j2ee课程设计—基于activiti的请休假系统的更多相关文章

  1. 嵌入式系统及应用课程设计——基于STM32的温湿度监测系统

    大三上学期期末总结,嗯,没错上学期,写在新学期开始,hhh. 上学期学了一门嵌入式系统及应用的课程,期末的课程设计题目是基于STM32的温湿度监测系统. 记得刚开始做课程设计的时候,听说先设计画出原理 ...

  2. 课程设计- 基于ssm的捐赠物资分配管理系统 && 基于java的申请救援管理系统

    课程设计- 基于ssm的捐赠物资分配管理系统 && 基于java的申请救援管理系统 注意:该项目只展示部分功能,如需了解,评论区咨询即可. 1.开发环境 开发语言:Java 后台框架: ...

  3. Java课程设计——博客作业教学数据分析系统(201521123084 林正晟)

    #课程设计--博客作业教学数据分析系统(201521123084 林正晟) 1.团队课程设计博客链接 博客作业教学数据分析系统 2.个人负责模块或任务说明 学生登陆界面的前端实现和与数据库的连接 学生 ...

  4. Java语言课程设计——博客作业教学数据分析系统(201521123107 张翔)

    #Java语言课程设计--博客作业教学数据分析系统(个人博客) 1.团队课程设计博客链接 [博客作业教学数据分析系统(From:网络五条狗)](http://www.cnblogs.com/fanta ...

  5. [课程设计]Scrum 多鱼点餐系统(团队交流日)

    [课程设计]Scrum  多鱼点餐系统(团队交流日) 1.团队名称:重案组 2.团队目标:长期经营,积累客户充分准备,伺机而行 3.团队口号:矢志不渝,追求完美 4.团队选题:餐厅到店点餐系统WEB ...

  6. Java课程设计——博客作业教学数据分析系统(201521123091 李嘉廉)

    #课程设计--博客作业教学数据分析系统(201521123084 李嘉廉) 1.团队课程设计博客链接 博客作业教学数据分析系统 2.个人负责模块或任务说明 數據分析 Kmeans聚類算法實現 多元綫性 ...

  7. Java课程设计——博客作业教学数据分析系统(201521123082 黄华林)

    Java课程设计--博客作业教学数据分析系统(201521123082 黄华林) 一.团队课程设计博客链接 博客作业教学数据分析系统(From:网络五条狗) 二.个人负责模块或任务说明 1.网络爬虫 ...

  8. Java团队课程设计——基于学院的搜索引擎

    团队名称.团队成员介绍.任务分配,团队成员课程设计博客链接 姓名 成员介绍 任务分配 课程设计博客地址 谢晓淞(组长) 团队输出主力 爬虫功能实现,Web前端设计及其后端衔接 爬虫:https://w ...

  9. java课程设计——博客作业教学数据分析系统(201521123083 戴志斌)

    目录 一.团队课程设计博客链接 二.个人负责模块或任务说明 三.自己的代码提交记录截图 四.自己负责模块或任务详细说明 五.课程设计感想 (题外话,终于可以用markdown建目录) 一.团队课程设计 ...

随机推荐

  1. SpringBoot 2.x.x会拦截静态资源问题

    之前在看尚硅谷视频时,可谓是按照这老师的说法,一步一步按部就班,于是采坑了,在SpringBoot 2.x.x之前是不会对静态资源拦截的,但是现在已经普通使用2.x.x这个版本会对静态资源进行拦截.看 ...

  2. 6、jeecg 笔记之 自定义excel 模板导出(一)

    1.前言 jeecg 中已经自带 excel 的导出导出功能,其所使用的是 easypoi,尽管所导出的 excel 能满足大部分需求, 但总是有需要用到自定义 excel 导出模板,下文所用到的皆是 ...

  3. Oracle中exp导出与imp导入的参数(full,owner/formuser/touser)测试

    1.exp导出的参数(FULL,OWNER)测试 先知道的一点是full不能与owner共存,还有都是以用户的方式导出(在这里),其中不仅仅包括表,这可能就是下面报warnings的原因,因为Orac ...

  4. iOS MJExtension的使用

    前言: MJExtension是iOS的字典装模型的一个第三方框架.相对于JSONKit和SBJson相比MJExtension更简单易用.功能更强大. 安装: 使用CocoaPods导入(Cocoa ...

  5. Python selenium中注入并执行Javascript语句

    众所周知,Python通常结合selenium模块来完成一些web的自动化测试以及RPA(Robotic Process Automation)工作.事实上,Selenium还可以支持插入js语句.执 ...

  6. Solaris环境下使用snoop命令抓包

    (1)报文抓取 Solaris中自带有snoop抓包工具,通过执行相应的命令抓取. 抓取目的地址为10.8.3.250的数据包,并存放到/opt/cap250的文件里 snoop -o /opt/ca ...

  7. c#中WebApi开发遇到的坑

    一.如何新建一个webApi项目 打开VS→找到解决方案→新建项目→类库或web应用程序→选择空的WebApi项目→在Global.asax文件的Application_Start方法中注册WebAp ...

  8. ASP.NET页面之间传值的方式之Cookie(个人整理)

    Cookie Cookie 提供了一种在 Web 应用程序中存储用户特定信息的方法.例如,当用户访问您的站点时,您可以使用 Cookie 存储用户首选项或其他信息.当该用户再次访问您的网站时,应用程序 ...

  9. 【SS & KCP centos7配置】

    1 依赖和环境配置 1.1 升级 yum $ yum update 1.2 安装 screen $ yum install screen 关于 screen 的详情:http://www.vpser. ...

  10. Delphi获取本机所有的IP

    安装Indy uses  IdStackWindows; var Isw:TIdStackWindows; slist:TStringList; begin Isw:=TIdStackWindows. ...