分类 未分类 下的文章

1:讲下Spr ingMvc和Struts1,Struts2的比较的优势

性能上Struts1>SpringMvc>Struts2 开发速度上SpringMvc和Struts2差不多,比Struts1要高

2:讲下SpringMvc的核心入口类是什么,Struts1,Struts2的分别是什么

SpringMvc的是DispatchServlet,Struts1的是ActionServlet,Struts2的是StrutsPrepareAndExecuteFilter

3:SpringMvc的控制器是不是单例模式,如果是,有什么问题,怎么解决

是单例模式,所以在多线程访问的时候有线程安全问题,不要用同步,会影响性能的,解决方案是在控制器里面不能写字段

4:SpingMvc中的控制器的注解一般用那个,有没有别的注解可以替代

一般用@Conntroller注解,表示是表现层,不能用用别的注解代替.

5:@RequestMapping注解用在类上面有什么作用

用于类上,表示类中的所有响应请求的方法都是以该地址作为父路径。

6:怎么样把某个请求映射到特定的方法上面

直接在方法上面加上注解@RequestMapping,并且在这个注解里面写上要拦截的路径

7:如果在拦截请求中,我想拦截get方式提交的方法,怎么配置

可以在@RequestMapping注解里面加上method=RequestMethod.GET

8:如果在拦截请求中,我想拦截提交参数中包含"type=test"字符串,怎么配置

可以在@RequestMapping注解里面加上params="type=test"

9:我想在拦截的方法里面得到从前台传入的参数,怎么得到

直接在形参里面声明这个参数就可以,但必须名字和传过来的参数一样

10: 如果前台有很多个参数传入,并且这些参数都是一个对象的,那么怎么样快速得到这个对象

直接在方法中声明这个对象,SpringMvc就自动会把属性赋值到这个对象里面

面试过程中SpringMVC被问到的知识点,你知道多少?

11:怎么样在方法里面得到Request,或者Session

直接在方法的形参中声明request,SpringMvc就自动把request对象传入

12:SpringMvc中函数的返回值是什么.

返回值可以有很多类型,有String, ModelAndView,当一般用String比较好

13:SpringMvc怎么处理返回值的

SpringMvc根据配置文件中InternalResourceViewResolver的前缀和后缀,用前缀+返回值+后缀组成完整的返回值

14:SpringMVC怎么样设定重定向和转发的

在返回值前面加"forward:"就可以让结果转发,譬如"forward:user.do?name=method4" 在返回值前面加"redirect:"就可以让返回值重定向,譬如"redirect://www.baidu.com"

15:SpringMvc用什么对象从后台向前台传递数据的

通过ModelMap对象,可以在这个对象里面用put方法,把对象加到里面,前台就可以通过el表达式拿到

16:SpringMvc中有个类把视图和数据都合并的一起的,叫什么

叫ModelAndView

17:怎么样把ModelMap里面的数据放入Session里面

可以在类上面加上@SessionAttributes注解,里面包含的字符串就是要放入session里面的key

18:SpringMvc怎么和AJAX相互调用的

通过Jackson框架就可以把Java里面的对象直接转化成Js可以识别的Json对象 具体步骤如下 1.加入Jackson.jar 2.在配置文件中配置json的映射 3.在接受Ajax方法里面可以直接返回Object,List等,但方法前面要加上@ResponseBody注解

19:当一个方法向AJAX返回特殊对象,譬如Object,List等,需要做什么处理

要加上@ResponseBody注解

20:SpringMvc里面拦截器是怎么写的

有两种写法,一种是实现接口,另外一种是继承适配器类,然后在SpringMvc的配置文件中配置拦截器即可: <!-- 配置SpringMvc的拦截器 --><mvc:interceptors> <!-- 配置一个拦截器的Bean就可以了 默认是对所有请求都拦截 --> <bean id="myInterceptor" class="com.et.action.MyHandlerInterceptor"></bean> <!-- 只针对部分请求拦截 --> <mvc:interceptor> <mvc:mapping path="/modelMap.do" /> <bean class="com.et.action.MyHandlerInterceptorAdapter" /> </mvc:interceptor></mvc:interceptors>

面试过程中SpringMVC被问到的知识点,你知道多少?

21:讲下SpringMvc的执行流程

系统启动的时候根据配置文件创建spring的容器, 首先是发送http请求到核心控制器disPatherServlet,spring容器通过映射器去寻找业务控制器,使用适配器找到相应的业务类,在进业务类时进行数据封装,在封装前可能会涉及到类型转换,执行完业务类后使用ModelAndView进行视图转发,数据放在model中,用map传递数据进行页面显示。

原文地址http://www.cnblogs.com/kimi9py/p/5697211.html

1 概要介绍

1)什么是流程定义 ?

流程定义是按照bpmn2.0标准定义业务流程,将流程定义的文件(.bpmn和.png(不是必须的))部署到activiti中,activiti就可以管理该业务流程。

 

2)什么是流程实例 ?

参与者(可以用户,也可以程序)按照流程定义发起一个流程,这个流程就是一个流程实例 。流程定义的内容就是一个静态文件(.bpmn),流程实例的内容是该 流程的执行过程(动态)。

 

3)如何启动一个流程实例 ?

常用方式:启动一个流程实例时指定一个业务标识。

    // 业务标识 ,如果 是采购流程就是采购单id

      String businessKey = "001";

      // 启动流程实例时指定业务标识

      String processDefinitionKey = "purchasingflow";

      ProcessInstance processInstance = runtimeService

            .startProcessInstanceByKey(processDefinitionKey, businessKey);

businessKey:业务标识,作用:通过activiti的api查询activiti的流程数据时,可以通过businessKey关联查询业务系统 的数据,通常业务标识记录业务系统表的主键,比如:如果采购流程,businesskey就是采购单id,如果是请假流程,businessKey就是请假单(请假信息表)的id。

 

启动一个流程实例 后,该流程运行到第一个结点,activiti需要给该 任务结点分配任务负责人。

 

任务分配三种方式:

 

第一种:采用固定分配方法,设置task结点的assignee(任务负责人)属性,不常用。

第二种:采用UEL表达式,表达使用流程变量设置任务负责人,在企业开发中常用。

第三种:采用监听器(自定义监听器实现 TaskListener接口),可以在监听器中扩展代码,在企业开发中常用。

 

查询待办任务:

查询待办任务中如果包括 业务系统 数据,如果查询?

通过Taskid得到任务所属的流程实例id

通过流程实例id得到流程实例对象ProcessInstance。

从ProcessInstance获取businessKey

通过businessKey关联查询业务数据

 

4)Activiti开发步骤:

1、  在需求阶段,分析出业务流程

2、  设计阶段,确定哪些业务流程由activiti管理,对工作流管理的流程进行流程定义

流程定义时和功能设计同步进行:

1》  确定流程启动所对应的功能

2》  确定流程执行中哪些功能和流程结点对应,哪个功能可以将流程向后推进一步

 

Activiti开发遵循原则:

1、  角色分工明确,activiti负责流程管理 ,业务系统 负责业务功能。

2、  业务系统 中通常在service层将activiti和控制层、持久层进行隔离(解耦),比如在业务功能中需要查询activiti的流程数据,需要自定义一个对象存储activiti的数据。

3、  数据共享问题,在activiti中存储businesskey(业务标识),通过businesskey查询业务系统 数据,在业务系统 中存储activiti的标识(比如在采购单中存储流程实例 的id),在查询业务数据时通过此流程实例 id查询activiti的数据。达到目标:activiti和业务系统 能互相关联查询。

办理任务(完成任务):

需要参数:任务id(activiti的api要求),用户id(进行权限校验使用)

在完成任务之前需要校验该 用户是否有该 任务的完成权限。

 

2 流程变量

2.1  什么是流程变量

在activiti在管理流程中,可能需要通过流程变量控制流程的执行。

注意:流程变量只是用于控制流程的执行,而不是存储业务数据!!

2.2  流程变量作用域

Activiti的流程变量包括 global全局变量和local局部变量。

Activiti常用全局变量进行流程控制,因为local局部变量作用域小一般不用,可以查询历史的局部变量。

global全局变量:作用域是整体流程实例,如果流程实例结束,变量无效。

local局部变量: 作用域小,可以是一个任务(task)也可以是一个执行分支(execution)。任务或执行分支结束,local局部变量无效。

通过historyService查询历史流程变量值。

不同的流程实例,global全局变量互不影响。

同一个流程实例,无法设置两个相同名称 global全局变量,后设置的相同名称 global全局变量会覆盖前边的变量值 。

任务中local变量,不同任务的local变量互不影响。Local变量名称可以和global全局变量相同。

2.3  流程变量的类型

注意:如果将pojo对象存储到流程变量中,必须实现序列化接口serializable,为了防止由于新增字段无法反序列化,需要生成serialVersionUID,如下:

注意:由于OrderCustom也继承了PurBusOrder类,PurBusOrder也必须实现序列化接口,否则 PurBusOrder中的属性无法反序列化。

总结:

1、如果需要将pojo中的属性从流程变量获取(activiti进行反序列化),需要将属性所在pojo实现serializable接口。

2、需要在pojo中设置serialVersionUID,如果不设置该版本id,日后在pojo中新增字段,如果之前将未新增字段的pojo对象存储在activiti的流程变量中,如果从流程变量中获取之前 的pojo对象,将报错。

2.4  流程变量的使用方法

第一步:设置流程变量

第二步:使用流程变量控制流程的执行

例子:

1) 在task结点的assignee通过UEL表达式使用流程变量

${assignee}:assignee就是一个流程变量名称

Activiti获取UEL表达式的值 ,即流程变量assignee的值 ,将assignee的值作为任务的负责人进行任务分配.

2)在连线上的candition条件上通过UEL表达式使用流程变量

${price>=10000}和${price<10000}: price就是一个流程变量名称,uel表达式结果类型为布尔类型

如果UEL表达式是true,要决定 流程执行走向。

比如:如果采购金额大于10000元由总经理审核,否则由财务直接审核。

2.5 设置全局变量

2.5.1 可以流程启动时设置流程变量(常用)

由于设置了全局变量,该流程启动后,在下边每个结点都可以使用该变量。

在task结点的assignee通过UEL表达式使用流程变量。

代码:

// 得到runtimeService

      RuntimeService runtimeService = processEngine.getRuntimeService();

      // 根据流程定义的key(标识 )来启动一个实例,activiti找该key下版本最高的流程定义

      // 一般情况下为了方便开发使用该方法启动一个流程实例。

      String processDefinitionKey = "purchasingflow";

      // 在流程启动时设置全局变量,第二个参数variables存储流程变量

      Map<String, Object> variables = new HashMap<String, Object>();

      // 流程 变更名称是assignee,流程变量值是"张三"

      variables.put("assignee", "张三");

      ProcessInstance processInstance = runtimeService.startProcessInstanceByKey(processDefinitionKey, variables);

2.5.2 在完成任务时设置流程变量(常用)

在完成任务时设置流程变量,在该 任务的后续结点可以使用该 变量,变量的作用域还是整个流程实例 。

//完成任务时设置流程变量

Map<String, Object> variables = new HashMap<String, Object>();

// 流程 变更名称是assignee,流程变量值是"张三"

variables.put("assignee", "李四");

taskService.complete(taskId, variables);

测试发现,在同一个流程实例下,同名的全局变量,后设置的会覆盖前边设置的。

2.5.3 通过当前流程实例 id设置

通过此方法可以在流程实例未结束时任意设置流程变量。

RuntimeService runtimeService  = processEngine.getRuntimeService();

      //设置单个 变量

      //第一个参数:executionId是流程实例 的执行 id,通常使用流程实例 id,必须是当前正在运行的流程实例 id

      //第二个参数:变量名

      //第三个参数:变量值

      runtimeService.setVariable("2001", "price", 10000);

      //一次设置多个变量,第二个参数是map

      //runtimeService.setVariables(executionId, variables)

注意:第一个参数executionId是流程实例 的执行 id,通常使用流程实例 id,必须是当前正在运行的流程实例 id

获取变量:

runtimeService.getVariable(executionId, variableName)

2.5.4 通过当前待办任务id设置

通过此方法可以在流程实例未结束时任意设置流程变量。

TaskService taskService = processEngine.getTaskService();

      //设置单个 变量

      //第一个参数:taskId是当前待办任务id,在act_ru_task存在

      //第二个参数:变量名

      //第三个参数:变量值

      //activiti根据 任务id查询所属流程实例 id,存储act_ru_variable表。

      taskService.setVariable("2902", "price", 10000);

      //一次设置多个变量,第二个参数是map

      //taskService.setVariables(taskId, variables)

 

activiti根据 任务id查询所属流程实例 id,将流程变量值存储act_ru_variable表,不管使用什么任务id,找到流程实例 id相同就会覆盖原来的变量。

获取变量方法:

System.out.println(taskService.getVariable("3302", "price"));

3  全局变量测试

什么时候设置流程变量?

采购流程中,在提交采购单时设置流程变量,因为提交采购单后采购单信息不再修改了。

使用完成提交采购单任务时设置流程变量。

// 完成任务时设置流程变量,使用pojo

         OrderCustom orderCustom = new OrderCustom();

         orderCustom.setPrice(10000f);

         Map<String, Object> varaibles = new HashMap<String, Object>();

         varaibles.put("order", orderCustom);

         taskService.complete(taskId, varaibles);

4  设置局部变量

4.1 在任务完成设置局部变量

在任务完成时设置局部变量,任务完成后,后续结点无法使用局部变量。

// 完成任务时设置流程变量,使用pojo

            OrderCustom orderCustom = new OrderCustom();

            orderCustom.setPrice(10000f);

            Map<String, Object> varaibles = new HashMap<String, Object>();

            varaibles.put("order", orderCustom);

            //设置局部变量

            taskService.setVariablesLocal(taskId, varaibles);

            //完成任务

            taskService.complete(taskId);

当局部变量消失后,再使用该变量会报错。

4.2 查询历史任务时查询流程变量

//查询历史 任务查询流程变量

historicTaskInstanceQuery.includeTaskLocalVariables();

//从activiti的流程变量取出pojo对象 ,经过反序列化

OrderCustom orderCustom = (OrderCustom) historicTaskInstance.getTaskLocalVariables().get("order");

         System.

out

.println("历史 流程变量采购单信息:"+orderCustom.getPrice());

注意:使用流程变量不建议存储业务数据,因为通过activiti的api查询流程变量,特别是pojo变量,速度很慢。

流程变量要用于流程控制。

4.3 通过当前任务id设置变量

//通过当前任务id设置局部变量

   @Test

   public void setVariableByTaskId(){

      TaskService taskService = processEngine.getTaskService();

      //设置单个 局部变量

      taskService.setVariableLocal("5305", "price", 50000);

      //一次设置多个局部变量

      //taskService.setVariablesLocal(taskId, variables)

   }

注意:任务id为当前待办理的任务id。

4.4  流程变量数据表跟踪

SELECT * FROM act_ru_variable #当前流程变量表

记录当前流程实例使用的流程变量

Type_:变量类型

Name_:变量名称

Execution_id_:流程实例执行id(global和local变量存储)

Proc_inst_id_:流程实例 id(global和local变量存储)

Task_id_:流程变量所属的任务id(local变量存储)

BYTEARRAY_ID_:如果流程变量为pojo,字段存储引用act_ge_bytearray表的主键,在资源表act_ge_bytearray存储pojo流程变量的序列化信息。

Long_和text_:根据变量类型存储变量值

SELECT * FROM act_hi_varinst #历史 流程变量表

记录流程执行所创建的所有流程变量

5 案例1

5.1 需求

员工创建采购单,由部门经理审核,部门经理审核通过后一万元以下由财务直接审核,一万元以上先由总经理审核,总经理审核通过再由财务审核。

5.2    流程定义

5.3    开发

采购流程中,在提交采购单时设置流程变量,因为提交采购单后采购单信息不再修改了。

使用完成提交采购单任务时设置流程变量。

修改提交 采购单service,在完成任务时设置流程变量。

修改orderService中saveOrderSubmitStauts提交采购单方法:

if (task != null) {

         // 说明assignee是该任务的办理人,有权限完成

         OrderCustom orderCustom = new OrderCustom();

         // 采购信息获取

         // 根据 任务对象 获取流程实例 id

         String processInstanceId = task.getProcessInstanceId();

         // 查询流程实例 对象

         ProcessInstance processInstance = runtimeService

                .createProcessInstanceQuery()

                .processInstanceId(processInstanceId).singleResult();

         // 从流程实例 对象 中获取businessKey

         String businessKey = processInstance.getBusinessKey();

         // 根据 businessKey查询采购单信息

         PurBusOrder purBusOrder = purBusOrderMapper.selectByPrimaryKey(businessKey);

         BeanUtils.copyProperties(purBusOrder, orderCustom);

         //流程变量,值 为orderCustom即采购单信息

         Map<String, Object> variables = new HashMap<String, Object>();

         variables.put("order", orderCustom);

         // 设置流程变量,值 为采购单信息

         taskService.complete(taskId,variables);

         // System.out.println("完成任务:" + taskId);

      }

6  案例2

6.1  需求

在上边的基础上添加审核不通过的连线。

需求如下:

在采购系统中实现流程审核不通过分支,功能如下:

1  部门经理审核不通过由员工重新修改采购单

2  总经理审核不通过由员工重新修改采购单再提交

3  财务审核不通过由员工重新审核

6.2   流程定义

6.2.1 对审核不通的由流程发起人再提交采购单

设置流程发起人:

在开始结点,设置initiator(发起人)。

设置创建采购单的任务负责人为:

修改orderService中的saveOrder方法,在启动流程时设置任务发起人:

注意:

任务发起人设置在启动流程实例之前设置。

6.2.2 增加审核不通过的连线

部门经理审核 :

 

审核通过candition:

${order.price>=10000 && firstAudit. status==’1’} 部门经理审核通过且采购金额大于等于10000元由总经理审核

${order.price<10000 && firstAudit. status==’1’}部门经理审核通过且采购金额小于10000元由财务审核

firstAudit和order都是流程变量名称

 

审核 不通过candition :

${ firstAudit. status==’0’}:部门经理审核不通过由流程发起人重新修改采购单再提交

 

总经理审核 :

审核通过candition:${ secondAudit. status==’1’} 总经理审核通过

审核 不通过candition : ${ secondAudit. status==’0’}总经理审核不通过

 

财务审核 :

审核通过candition:${ thirdAudit. status==’1’} 财务审核通过

审核 不通过candition : ${ third Audit. status==’0’}财务审核不通过

注意:上边的firstAudit、secondAudit、thirdAudit分别存储三级审核信息。

6.3  开发

修改orderService中saveOrderSubmitStauts提交采购单方法:

if (task != null) {

         // 说明assignee是该任务的办理人,有权限完成

         OrderCustom orderCustom = new OrderCustom();

         // 采购信息获取

         // 根据 任务对象 获取流程实例 id

         String processInstanceId = task.getProcessInstanceId();

         // 查询流程实例 对象

         ProcessInstance processInstance = runtimeService

                .createProcessInstanceQuery()

                .processInstanceId(processInstanceId).singleResult();

         // 从流程实例 对象 中获取businessKey

         String businessKey = processInstance.getBusinessKey();

         // 根据 businessKey查询采购单信息

         PurBusOrder purBusOrder = purBusOrderMapper.selectByPrimaryKey(businessKey);

         BeanUtils.copyProperties(purBusOrder, orderCustom);

         //流程变量,值 为orderCustom即采购单信息

         Map<String, Object> variables = new HashMap<String, Object>();

         variables.put("order", orderCustom);

         // 设置流程变量,值 为采购单信息

         taskService.complete(taskId,variables);

         // System.out.println("完成任务:" + taskId);

      }

 

修改 service中采购单审核方法:

设置流程变量:firstAudit、secondAudit、thirdAudit

 

if (task != null) {

         // 说明assignee是该任务的办理人,有权限完成

         Map<String,Object> variables = new HashMap<String,Object>();

         //根据 auditType判断是几级审核

         if(auditType.equals("firstAudit")){

            //部门经理审核

            variables.put("firstAudit", orderAuditCustom);

         }else if(auditType.equals("secondAudit")){

            //总经理审核

            variables.put("secondAudit", orderAuditCustom);

         }else if(auditType.equals("thirdAudit")){

            //财务审核

            variables.put("thirdAudit", orderAuditCustom);

         }

         //提交审核时,设置流程变量,变量值就是审核 信息

         taskService.complete(taskId,variables);

         // System.out.println("完成任务:" + taskId);

      }

 

7  Candidate-user候选人

7.1  什么是候选人

采用固定分配方法给任务指定负责人,如果任务负责人出现变更,需要修改流程定义,就可以采用候选人分配方式,先给任务分配多个候选人,候选人通过拾取组任务进行个人任务办理。

给任务分配候选人,如果分配多个候选人中间使用半角逗号分隔。

7.2  什么是组任务

多个候选人有资格完成该 任务,这个任务叫做组任务。

组任务具备条件:

任务没有设置assignee任务负责人

任务具有候选人

7.3   候选人办理任务过程

第一步:给任务设置候选人(多个, 中间使用半角逗号分隔)

候选人是无法办理任务

第二步:候选人查询组任务

使用taskService查询,指定candidate候选人。

第三步:候选人拾取(claim)组任务

候选人拾取组任务后,该 候选人变为任务的负责人,该任务变为个人任务

如果候选人拾取组任务后,不想办理该 任务,可以将个人任务归还,该个人任务变为组任务

第四步:查询待办个人任务

第五步:办理任务

第六步:流程结束

7.4   Candidate-user办理任务api

7.4.1 候选人查询组任务

使用taskService指定candidateUser候选人查询组任务。

//任务查询对象

      TaskQuery taskQuery = taskService.createTaskQuery();

      //候选人

      String candidateUser = "zhangsan";

      taskQuery.taskCandidateUser(candidateUser);

      //流程定义key

      String processDefinitionKey = "purchasingflow";

      taskQuery.processDefinitionKey(processDefinitionKey);

      List<Task> list = taskQuery.list();

注意:查询组任务,必须指定 candidateUser候选人,查询该候选人有资格办理的组任务。

7.4.2 拾取组任务

通过taskService,指定任务id和候选人拾取任务:

TaskService taskService = processEngine.getTaskService();

      //组任务id

      String taskId = "5604";

      //任务候选人,claim拾取后该 候选人变为任务负责人

      String userId = "zhangsan";

      //任务拾取

      taskService.claim(taskId, userId);

注意:如果拾取人不是该任务的候选人也可以拾取成功,在拾取之前需要校验,该 候选人是否有资格拾取该 任务.

// 组任务id

      String taskId = "6004";

      // 任务候选人,claim拾取后该 候选人变为任务负责人

      String candidateUser = "zhangsan4";

      //根据候选人和组任务id查询,如果有记录说明该 候选人有资格拾取该 任务

      Task task = taskService.createTaskQuery().taskId(taskId)

            .taskCandidateUser(candidateUser).singleResult();

      if(task!=null){

         // 任务拾取

         taskService.claim(taskId, candidateUser);

         System.out.println("任务拾取成功");

      }

7.4.3 组任务归还

// 归还组任务,由个人任务变为组任务,还可以进行任务交接

   @Test

   public void setAssignee() {

      // 查询任务使用TaskService

      TaskService taskService = processEngine.getTaskService();

      // 当前待办任务

      String taskId = "6004";

      // 任务负责人

      String userId = "zhangsan2";

      //校验userId是否是taskId的负责人,如果是负责人才可以归还组任务

      Task task = taskService.createTaskQuery().taskId(taskId).taskAssignee(userId).singleResult();

      if(task!=null){

         //如果设置为null,归还组任务,该 任务没有负责人

         taskService.setAssignee(taskId, null);

      }

   }

7.4.4 任务交接

任务负责人也可以将任务交给其它候选人办理该任务

代码如下:

@Test

   public void setAssigneeToCandidateUser() {

      // 查询任务使用TaskService

      TaskService taskService = processEngine.getTaskService();

      // 当前待办任务

      String taskId = "6004";

      // 任务负责人

      String userId = "zhangsan2";

      // 校验userId是否是taskId的负责人,如果是负责人才可以归还组任务

      Task task = taskService.createTaskQuery().taskId(taskId)

            .taskAssignee(userId).singleResult();

      if (task != null) {

         // 将此任务交给其它候选人办理该 任务

         String candidateuser = "zhangsan";

         // 根据候选人和组任务id查询,如果有记录说明该 候选人有资格拾取该 任务

         Task task2 = taskService.createTaskQuery().taskId(taskId)

                .taskCandidateUser(candidateuser).singleResult();

         if (task2 != null) {

            // 才可以交接

            taskService.setAssignee(taskId, candidateuser);

         }

      }

   }

8  Candidate-group候选组

8.1   什么候选组

即使给任务指定了多个候选人,多个候选人都有办理任务资格,但是候选的人数有限,无法动态扩展,如果需要添加或删除候选,需要修改流程定义 文件,不利于系统 扩展。

采用候选组方式解决上边的问题。

给任务设置候选组,在组中有多个用户并且可以动态扩展用户,组中的用户都是候选人,候选人先拾取组任务,将组任务变为自己的个人任务,进行个人任务办理。

8.2   候选组办理任务过程

第一步:Activiti会自动从候选组中找用户,将这些用户作为该 任务的候选人。

下边的流程同候选人办理任务过程!!

第二步:给任务设置候选人(多个, 中间使用半角逗号分隔)

候选人是无法办理任务

第三步:候选人查询组任务

使用taskService查询,指定candidate候选人。

第四步:候选人拾取(claim)组任务

候选人拾取组任务后,该 候选人变为任务的负责人,该任务变为个人任务

如果候选人拾取组任务后,不想办理该 任务,可以将个人任务归还,该个人任务变为组任务

第五步:查询待办个人任务

第六步:办理任务

第七步:流程结束

8.3  设置候选组

8.4  设置组和用户信息

Activiti中采用以下表记录组信息、用户信息、组和用户关系 信息

SELECT * FROM act_id_group #组信息

SELECT * FROM act_id_user #用户信息

SELECT * FROM act_id_membership #组和用户关系信息

8.4.1 Api设置方法

以下设置的信息和业务系统的用户信息、角色信息保存一致。

先设置组信息

再设置用户信息

再设置组和用户关系信息

代码如下:

//设置组和用户信息

   @Test

   public void setUserGroup(){

      IdentityService identityService = processEngine.getIdentityService();

      //设置组信息

      //添加之前应该校验组信息是否存在,不存在再进行添加

      if(identityService.createGroupQuery().groupId("10").singleResult()==null){

         //添加新组

         GroupEntity groupEntity = new GroupEntity();

         groupEntity.setId("10");

         groupEntity.setName("员工");

         identityService.saveGroup(groupEntity);

      }

      if(identityService.createGroupQuery().groupId("11").singleResult()==null){

         //添加新组

         GroupEntity groupEntity = new GroupEntity();

         groupEntity.setId("11");

         groupEntity.setName("部门经理");

         identityService.saveGroup(groupEntity);

      }

      if(identityService.createGroupQuery().groupId("12").singleResult()==null){

         //添加新组

         GroupEntity groupEntity = new GroupEntity();

         groupEntity.setId("12");

         groupEntity.setName("总经理");

         identityService.saveGroup(groupEntity);

      }

      if(identityService.createGroupQuery().groupId("13").singleResult()==null){

         //添加新组

         GroupEntity groupEntity = new GroupEntity();

         groupEntity.setId("13");

         groupEntity.setName("财务");

         identityService.saveGroup(groupEntity);

      }

      //设置用户信息

      //添加之前应该校验用户信息是否存在,不存在再进行添加

      if(identityService.createUserQuery().userId("zhangsan").singleResult()==null){

         //添加新用户

         UserEntity userEntity = new UserEntity();

         userEntity.setId("zhangsan");

         userEntity.setFirstName("张三");

         identityService.saveUser(userEntity);

      }

      if(identityService.createUserQuery().userId("lisi").singleResult()==null){

         //添加新用户

         UserEntity userEntity = new UserEntity();

         userEntity.setId("lisi");

         userEntity.setFirstName("李四");

         identityService.saveUser(userEntity);

      }

      if(identityService.createUserQuery().userId("wangwu").singleResult()==null){

         //添加新用户

         UserEntity userEntity = new UserEntity();

         userEntity.setId("wangwu");

         userEntity.setFirstName("王五");

         identityService.saveUser(userEntity);

      }

      if(identityService.createUserQuery().userId("zhaoliu").singleResult()==null){

         //添加新用户

         UserEntity userEntity = new UserEntity();

         userEntity.setId("zhaoliu");

         userEntity.setFirstName("赵六");

         identityService.saveUser(userEntity);

      }

      //设置用户和组的关系信息

      //采用先删除再添加

      identityService.deleteMembership("zhangsan", "10");

      identityService.createMembership("zhangsan", "10");

      identityService.deleteMembership("lisi", "11");

      identityService.createMembership("lisi", "11");

      identityService.deleteMembership("wangwu", "12");

      identityService.createMembership("wangwu", "12");

      identityService.deleteMembership("zhaoliu", "13");

      identityService.createMembership("zhaoliu", "13");

   }

 

8.4.2 与业务系统同步方法(常用)

方法1 :

数据库触发器方法

企业中在进行数据同步的常用方法,一般在一个数据库中采用此方法。

业务系统 用户表----》activiti的act_id_user

在业务系统 用户表添加触发器:新增、删除、修改

注意:如果act_id_user有外键关系,需要先删除依赖关系。

业务系统 角色表----》activiti的act_id_group

业务系统 角色和用户关系表—》activiti的act_id_membership

 

方法2 :

采用即时触发java程序。

用户角色同步:

在操作业务系统 用户角色表时执行以下操作:

业务系统添加角色-àactiviti添加角色

业务系统修改角色àactiviti修改角色

业务系统删除角色àactiviti删除角色,删除之前将用户角色关系表先删除(根据角色删除)

 

用户信息同步

在操作业务系统 用户表时执行以下操作

业务系统添加用户----》activiti添加用户,添加用户与角色关系表

业务系统修改用户---》activiti修改用户,先删除原来用户与角色关系表,再添加用户与角色关系

业务系统删除用户—》activiti删除用户,删除之前将用户角色关系表删除(根据用户删除)

8.5  组任务办理过程api

8.5.1 设置组和用户信息

参考上边设置组和用户api

正式开发时,需要将业务系统 用户和角色信息同步到activiti中。

8.5.2 候选人查询组任务

参考candidate-user的api

注意:

在activiti的用户、组、用户和组关系表中随时添加数据,不受流程启动先后顺序影响。

9  网关

9.1   排他网关

9.1.1 什么排他网关

排他网关用于决策,选择分支执行流程,分支上需要设置condition条件,如果分支的条件结果为true,那么该分支会通过排他网关。排他网关只会选择一条分支去执行。

9.1.2 定义方法

图标:

流程定义:

 

 

9.1.3 排他网关测试

第一步:流程定义部署

第二步:启动流程实例

设置price流程变量值,因为price 在排他网关的两分支使用

第三步:查询待办任务

也可以在部门经理审核后设置price流程变量值,因为price 在排他网关的两分支使用

第四步:办理任务

如果分支上的条件都不满足,没有一条线经过排他网关,activiti会抛出异常:

org.activiti.engine.ActivitiException: No outgoing sequence flow of the exclusive gateway 'exclusivegateway1' could be selected for continuing the process at org.activiti.engine.impl.bpmn.behavior.ExclusiveGatewayActivityBehavior.leave(ExclusiveGatewayActivityBehavior.java:85)

如果多条分支都满足,只会有一条线经过排他网关。

上边两种情况必须在开发避免!!!

9.2  并行网关

9.2.1 什么并行网关

并行网关(parallelGateway),包括分支和汇聚两个结点,所有的分支不判断条件都经过分支结点,所有经过分支结点的分支都要进行汇聚,所有的分支全部执行完成,并行网关执行完成。

Fork(分支)

所有的分支不判断条件都经过分支结点

Join(汇聚)

所有经过分支结点的分支都要进行汇聚

分支的数量等于汇聚数量!

9.2.2 流程定义

图标:

 

 

 

注意:经过并行网关的分支结点,不需要设置condition条件。

9.2.3 并行网关测试

当流程执行到并行的分支结点时,

向act_ru_execution #流程实例执行表执行并行分支(结算,入库)

Execution表中8501的记录数等于分支数+1

只有一条记录的流程实例 id和流程实例执行id相等的,这一条为流程执行主线。

向当前任务表中插入两条记录(结算、入库)

9.3   包含网关

9.3.1 什么是包含网关

包含网关是排他网关和并行网关的结合体。

包含网关(IncluesiveGateway),包括分支和汇聚两个结点,经过分支结点需要判断条件,满足条件经过分支结点,所有经过分支结点的线边最终会进行汇聚。

Fork(分支)

所有的分支需要判断条件,满足条件的经过分支结点

Join(汇聚)

所有满足条件的分支都要进行汇聚

10   案例

10.1      需求

将采购流程改为组任务(使用候选组)实现

在采购流程中实现排他网关

在采购流程中实现并行网关

 

需求描述:

员工创建采购单

经过部门经理审核

审核通过:

部门经理审核通过,如果采购金额大于等于1万元,由总经理审核

部门经理审核通过,如果采购金额小于1万元,由财务审核

审核不通过:

部门经理审核不通过,由员工重新修改采购单进行提交

 

总经理审核

总经理审核通过由财务审核通过

总经理审核不通过,由员工重新修改采购单进行提交

 

财务审核

财务审核通过并行执行财务结算和入库

财务审核不通过,由员工重新修改采购单进行提交

 

财务结算和入库两个操作可以并行执行。

10.2   流程定义

部门经理审核通过后,通过排他网关决定走总经理审核还是财务审核。

财务审核通过后,经过并行网关,财务结算和入库并行执行。

 

审核分支设置condition 条件:

部门经理审核 :

 

审核通过candition:

${order.price>=10000 && firstAudit. status==’1’} 部门经理审核通过且采购金额大于等于10000元由总经理审核

${order.price<10000 && firstAudit. status==’1’}部门经理审核通过且采购金额小于10000元由财务审核

firstAudit和order都是流程变量名称

 

审核 不通过candition :

${ firstAudit. status==’0’}:部门经理审核不通过由流程发起人重新修改采购单再提交

 

总经理审核 :

审核通过candition:${ secondAudit. status==’1’} 总经理审核通过

审核 不通过candition : ${ secondAudit. status==’0’}总经理审核不通过

 

财务审核 :

审核通过candition:${ thirdAudit. status==’1’} 财务审核通过

审核 不通过candition : ${ third Audit. status==’0’}财务审核不通过

10.3  分析

10.3.1 需要开发结算和入库功能

真正的结算和入库很复杂,确定一个功能将流程向后推进一步。

为了教学方便,开发结算和入库功能实现activiti任务完成。

10.3.2 候选人查询组任务

由于任务结点改变  候选组方式分配任务,实现候选人查询组任务。

实现 方法:

调用taskService,指定candidateUser候选人查询组任务。

10.4   开发

接口功能:候选人查询组任务

接口参数:候选人candidateUserId,(实际开发需要查询条件)

接口实现 :

调用taskService,指定candidateUser候选人查询组任务

public List<OrderCustom> findOrderGroupTaskList(String userId)

         throws Exception {

      // 任务查询对象

      TaskQuery taskQuery = taskService.createTaskQuery();

      // 候选人,在act_id_user表中存在,从act_id_membership通过group_id_查询出用户

      String candidateUser = userId;

      // 指定候选人

      taskQuery.taskCandidateUser(candidateUser);

      // 流程定义key

      String processDefinitionKey = ResourcesUtil.getValue(

            "diagram.purchasingflow", "purchasingProcessDefinitionKey");

      taskQuery.processDefinitionKey(processDefinitionKey);

      List<Task> list = taskQuery.list();

      List<OrderCustom> orderList = new ArrayList<OrderCustom>();

      for (Task task : list) {

         OrderCustom orderCustom = new OrderCustom();

         //下边的代码同采购单处理列表代码...

         // 流程实例id

         String processInstanceId = task.getProcessInstanceId();

         // 根据流程实例id找到流程实例对象

         ProcessInstance processInstance = runtimeService

                .createProcessInstanceQuery()

                .processInstanceId(processInstanceId).singleResult();

         // 从流程实例对象中获取businessKey

         String businessKey = processInstance.getBusinessKey();

         // 根据businessKey查询业务系统

         // 采购单id

         String orderId = businessKey;

         PurBusOrder purBusOrder = purBusOrderMapper

                .selectByPrimaryKey(orderId);

         // 获取采购单名称、采购金额等采购单信息

         // 将purBusOrder内容拷贝到orderCustom

         BeanUtils.copyProperties(purBusOrder, orderCustom);

         // 下边向orderCustom开始设置任务信息

         // 任务id、任务标识 、任务名称

         // 任务id

         orderCustom.setTaskId(task.getId());

         // 任务标识

         orderCustom.setTaskDefinitionKey(task.getTaskDefinitionKey());

         // 任务名称

         orderCustom.setTaskName(task.getName());

         orderList.add(orderCustom);

      }

      return orderList;

   }

 

查询组任务方法:

代码基本上同采购单处理列表的代码:

 

// 采购单组任务列表

   @RequestMapping("/orderGroupTaskList")

   public String orderGroupTaskList(HttpSession session, Model model)

         throws Exception {

      // 当前登陆用户

      ActiveUser activeuser = UserUtil.getUserFromSession(session);

      // 用户id

      String userId = activeuser.getUserid();

      List<OrderCustom> list = orderService.findOrderGroupTaskList(userId);

      model.addAttribute("list", list);

      return "order/orderGroupTaskList";

   }

 

接口功能:拾取组任务

接口参数:taskId任务id,candidateUserId候选人

接口实现 :

调用taskService,指定组任务id和candidateUserId候选人。

拾取任务之前需要校验候选人是否资格拾取该组任务。

 

 

 

@Override

   public void saveClaimTask(String taskId, String candidateUserId)

         throws Exception {

      // 根据候选人和组任务id查询,如果有记录说明该 候选人有资格拾取该 任务

      Task task = taskService.createTaskQuery().taskId(taskId)

            .taskCandidateUser(candidateUserId).singleResult();

      if (task != null) {

         // 任务拾取

         taskService.claim(taskId, candidateUserId);

         System.out.println("任务拾取成功");

      }

   }

 

拾取组任务方法:

需要从页面传入taskId组任务id.

 

// 拾取组任务

   @RequestMapping("/claimTask")

   public String claimTask(HttpSession session, String taskId)

         throws Exception {

      // 当前登陆用户

      ActiveUser activeuser = UserUtil.getUserFromSession(session);

      // 用户id

      String userId = activeuser.getUserid();

      orderService.saveClaimTask(taskId, userId);

      //返回采购单组任务列表

      return "redirect:orderGroupTaskList.action";

   }

 

 

修改组任务列表页面,添加“拾取组任务”连接:

 

1.2      测试

 

测试注意点:

1、组任务(使用候选组)

准备组和用户的数据,调用activiti的api设置组、用户、组和用户关系信息

正式开发,将业务系统 用户角色数据实时同步到activiti中。

 

2、测试组任务办理流程

a>查询组任务功能

b>拾取组任务功能

 

3、测试排他网关

修改原来的功能是否存在bug。

4、测试并行网关

结算功能和入库功能是否并行执行.

 

11  总结

什么是工作流?

工作流是通过计算机自动管理业务流程,实现多个参与者按照预定义的流程自动执行业务流程。

 

什么是activiti?

Activiti是一个工作流的引擎(框架,jar、组件),对业务流程的自动化管理。Activiti按照bpmn2.0标准进行流程定义,按照定义流程(bpmn文件) 去自动执行业务流程。

 

 

第一步:线下进行流程定义

Activiti按照bpmn2.0标准进行流程定义,定义文件包括 .bpmn和.png,其中.bpmn是必须的文件。

第二步:进行流程定义部署

将线下流程定义文件.bpmn部署到activiti的数据库中。这样activiti方可按照定义流程(bpmn文件) 去自动执行业务流程。

两种方法部署方法:

1、  单个文件部署方法

2、  Zip包部署方法

建议使用单个文件 部署,建议一次部署只部署一个流程定义 。

第三步:启动一个流程 实例

 

流程定义和流程实例 的区别:

流程定义 是一个静态文件(.bpmn),流程实例是动态的流程执行过程。如果要activiti去管理业务流程的执行,需要首先发起一个流程实例 。

 

第四步:查询待办任务

 

个人任务:

通过固定设置方法或UEL表达式设置方法设置task结点的assignee属性。

 

组任务:

通过设置多个候选人或多个候选组的方法。

组任务常用,组任务中以candidate-group候选组方式最常用。

 

组任务办理过程:

通过候选人查询任务

拾取任务

同个人任务办理流程。

 

第五步:办理任务

 

流程的执行可能需要通过流程变量决定执行分支。

流程变量:

常用全局变量

需要在启动流程实例 或完成任务时设置流程变量

在UEL表达式中使用流程变量

 

网关:

排他网关:

经过排他网关的分支只有一条。

并行网关:

经过并行网关分支结点所有分支不管条件是否满足都 会经过并行网关的汇聚结点。

并行网关常用于会签任务(多个用户共同办理的任务)

包含网关:

经过包含网关分支结点,只有满足条件的分支才经过包含网关的汇聚结点。

第六步:流程完成

所有结点完成执行完成,流程自动完成。

注意:如果使用并行网关或包含网关,必须经过分支的流程全部执行完成才能到达汇聚结点,并行网关或包含风关就结束。

Spring 概述

1. 什么是spring?

Spring 是个java企业级应用的开源开发框架。Spring主要用来开发Java应用,但是有些扩展是针对构建J2EE平台的web应用。Spring 框架目标是简化Java企业级应用开发,并通过POJO为基础的编程模型促进良好的编程习惯。

2. 使用Spring框架的好处是什么?

轻量:Spring 是轻量的,基本的版本大约2MB。

控制反转:Spring通过控制反转实现了松散耦合,对象们给出它们的依赖,而不是创建或查找依赖的对象们。

面向切面的编程(AOP):Spring支持面向切面的编程,并且把应用业务逻辑和系统服务分开。

容器:Spring 包含并管理应用中对象的生命周期和配置。

MVC框架:Spring的WEB框架是个精心设计的框架,是Web框架的一个很好的替代品。

事务管理:Spring 提供一个持续的事务管理接口,可以扩展到上至本地事务下至全局事务(JTA)。

异常处理:Spring 提供方便的API把具体技术相关的异常(比如由JDBC,Hibernate or JDO抛出的)转化为一致的unchecked 异常。

3. Spring由哪些模块组成?

以下是Spring 框架的基本模块:

Core module

Bean module

Context module

Expression Language module

JDBC module

ORM module

OXM module

Java Messaging Service(JMS) module

Transaction module

Web module

Web-Servlet module

Web-Struts module

Web-Portlet module

4. 核心容器(应用上下文) 模块

这是基本的Spring模块,提供spring 框架的基础功能,BeanFactory 是 任何以spring为基础的应用的核心。Spring 框架建立在此模块之上,它使Spring成为一个容器。

5. BeanFactory – BeanFactory 实现举例

Bean 工厂是工厂模式的一个实现,提供了控制反转功能,用来把应用的配置和依赖从正真的应用代码中分离。

最常用的BeanFactory 实现是XmlBeanFactory 类。

6. XMLBeanFactory

最常用的就是org.springframework.beans.factory.xml.XmlBeanFactory ,它根据XML文件中的定义加载beans。该容器从XML 文件读取配置元数据并用它去创建一个完全配置的系统或应用。

7. 解释AOP模块

AOP模块用于发给我们的Spring应用做面向切面的开发, 很多支持由AOP联盟提供,这样就确保了Spring和其他AOP框架的共通性。这个模块将元数据编程引入Spring。

8. 解释JDBC抽象和DAO模块

通过使用JDBC抽象和DAO模块,保证数据库代码的简洁,并能避免数据库资源错误关闭导致的问题,它在各种不同的数据库的错误信息之上,提供了一个统一的异常访问层。它还利用Spring的AOP 模块给Spring应用中的对象提供事务管理服务。

9. 解释对象/关系映射集成模块

Spring 通过提供ORM模块,支持我们在直接JDBC之上使用一个对象/关系映射映射(ORM)工具,Spring 支持集成主流的ORM框架,如Hiberate,JDO和 iBATIS SQL Maps。Spring的事务管理同样支持以上所有ORM框架及JDBC。

10. 解释WEB 模块

Spring的WEB模块是构建在application context 模块基础之上,提供一个适合web应用的上下文。这个模块也包括支持多种面向web的任务,如透明地处理多个文件上传请求和程序级请求参数的绑定到你的业务对象。它也有对Jakarta Struts的支持。

11. Spring配置文件

Spring配置文件是个XML 文件,这个文件包含了类信息,描述了如何配置它们,以及如何相互调用。

12. 什么是Spring IOC 容器?

Spring IOC 负责创建对象,管理对象(通过依赖注入(DI),装配对象,配置对象,并且管理这些对象的整个生命周期。

13. IOC的优点是什么?

IOC 或 依赖注入把应用的代码量降到最低。它使应用容易测试,单元测试不再需要单例和JNDI查找机制。最小的代价和最小的侵入性使松散耦合得以实现。IOC容器支持加载服务时的饿汉式初始化和懒加载。

14. ApplicationContext通常的实现是什么?

FileSystemXmlApplicationContext :此容器从一个XML文件中加载beans的定义,XML Bean 配置文件的全路径名必须提供给它的构造函数。

ClassPathXmlApplicationContext:此容器也从一个XML文件中加载beans的定义,这里,你需要正确设置classpath因为这个容器将在classpath里找bean配置。

WebXmlApplicationContext:此容器加载一个XML文件,此文件定义了一个WEB应用的所有bean。

15. Bean 工厂和 Application contexts 有什么区别?

Application contexts提供一种方法处理文本消息,一个通常的做法是加载文件资源(比如镜像),它们可以向注册为监听器的bean发布事件。另外,在容器或容器内的对象上执行的那些不得不由bean工厂以程序化方式处理的操作,可以在Application contexts中以声明的方式处理。Application contexts实现了MessageSource接口,该接口的实现以可插拔的方式提供获取本地化消息的方法。

16. 一个Spring的应用看起来象什么?

一个定义了一些功能的接口。

这实现包括属性,它的Setter , getter 方法和函数等。

Spring AOP。

Spring 的XML 配置文件。

使用以上功能的客户端程序。

依赖注入

17. 什么是Spring的依赖注入?

依赖注入,是IOC的一个方面,是个通常的概念,它有多种解释。这概念是说你不用创建对象,而只需要描述它如何被创建。你不在代码里直接组装你的组件和服务,但是要在配置文件里描述哪些组件需要哪些服务,之后一个容器(IOC容器)负责把他们组装起来。

18. 有哪些不同类型的IOC(依赖注入)方式?

构造器依赖注入:构造器依赖注入通过容器触发一个类的构造器来实现的,该类有一系列参数,每个参数代表一个对其他类的依赖。

Setter方法注入:Setter方法注入是容器通过调用无参构造器或无参static工厂 方法实例化bean之后,调用该bean的setter方法,即实现了基于setter的依赖注入。

19. 哪种依赖注入方式你建议使用,构造器注入,还是 Setter方法注入?

你两种依赖方式都可以使用,构造器注入和Setter方法注入。最好的解决方案是用构造器参数实现强制依赖,setter方法实现可选依赖。

Spring Beans

20.什么是Spring beans?

Spring beans 是那些形成Spring应用的主干的java对象。它们被Spring IOC容器初始化,装配,和管理。这些beans通过容器中配置的元数据创建。比如,以XML文件中<bean/> 的形式定义。

Spring 框架定义的beans都是单件beans。在bean tag中有个属性”singleton”,如果它被赋为TRUE,bean 就是单件,否则就是一个 prototype bean。默认是TRUE,所以所有在Spring框架中的beans 缺省都是单件。

21. 一个 Spring Bean 定义 包含什么?

一个Spring Bean 的定义包含容器必知的所有配置元数据,包括如何创建一个bean,它的生命周期详情及它的依赖。

22. 如何给Spring 容器提供配置元数据?

这里有三种重要的方法给Spring 容器提供配置元数据。

XML配置文件。

基于注解的配置。

基于java的配置。

23. 你怎样定义类的作用域?

当定义一个<bean> 在Spring里,我们还能给这个bean声明一个作用域。它可以通过bean 定义中的scope属性来定义。如,当Spring要在需要的时候每次生产一个新的bean实例,bean的scope属性被指定为prototype。另一方面,一个bean每次使用的时候必须返回同一个实例,这个bean的scope 属性 必须设为 singleton。

24. 解释Spring支持的几种bean的作用域。

Spring框架支持以下五种bean的作用域:

singleton : bean在每个Spring ioc 容器中只有一个实例。

prototype:一个bean的定义可以有多个实例。

request:每次http请求都会创建一个bean,该作用域仅在基于web的Spring ApplicationContext情形下有效。

session:在一个HTTP Session中,一个bean定义对应一个实例。该作用域仅在基于web的Spring ApplicationContext情形下有效。

global-session:在一个全局的HTTP Session中,一个bean定义对应一个实例。该作用域仅在基于web的Spring ApplicationContext情形下有效。

缺省的Spring bean 的作用域是Singleton。

25. Spring框架中的单例bean是线程安全的吗?

不,Spring框架中的单例bean不是线程安全的。

26. 解释Spring框架中bean的生命周期。

Spring容器 从XML 文件中读取bean的定义,并实例化bean。

Spring根据bean的定义填充所有的属性。

如果bean实现了BeanNameAware 接口,Spring 传递bean 的ID 到 setBeanName方法。

如果Bean 实现了 BeanFactoryAware 接口, Spring传递beanfactory 给setBeanFactory 方法。

如果有任何与bean相关联的BeanPostProcessors,Spring会在postProcesserBeforeInitialization()方法内调用它们。

如果bean实现IntializingBean了,调用它的afterPropertySet方法,如果bean声明了初始化方法,调用此初始化方法。

如果有BeanPostProcessors 和bean 关联,这些bean的postProcessAfterInitialization() 方法将被调用。

如果bean实现了 DisposableBean,它将调用destroy()方法。

27. 哪些是重要的bean生命周期方法? 你能重载它们吗?

有两个重要的bean 生命周期方法,第一个是setup , 它是在容器加载bean的时候被调用。第二个方法是 teardown 它是在容器卸载类的时候被调用。

The bean 标签有两个重要的属性(init-method和destroy-method)。用它们你可以自己定制初始化和注销方法。它们也有相应的注解(@PostConstruct和@PreDestroy)。

28. 什么是Spring的内部bean?

当一个bean仅被用作另一个bean的属性时,它能被声明为一个内部bean,为了定义inner bean,在Spring 的 基于XML的 配置元数据中,可以在 <property/>或 <constructor-arg/> 元素内使用<bean/> 元素,内部bean通常是匿名的,它们的Scope一般是prototype。

29. 在 Spring中如何注入一个java集合?

Spring提供以下几种集合的配置元素:

<list>类型用于注入一列值,允许有相同的值。

<set> 类型用于注入一组值,不允许有相同的值。

<map> 类型用于注入一组键值对,键和值都可以为任意类型。

<props>类型用于注入一组键值对,键和值都只能为String类型。

30. 什么是bean装配?

装配,或bean 装配是指在Spring 容器中把bean组装到一起,前提是容器需要知道bean的依赖关系,如何通过依赖注入来把它们装配到一起。

31. 什么是bean的自动装配?

Spring 容器能够自动装配相互合作的bean,这意味着容器不需要<constructor-arg>和<property>配置,能通过Bean工厂自动处理bean之间的协作。

32. 解释不同方式的自动装配 。

有五种自动装配的方式,可以用来指导Spring容器用自动装配方式来进行依赖注入。

no:默认的方式是不进行自动装配,通过显式设置ref 属性来进行装配。

byName:通过参数名 自动装配,Spring容器在配置文件中发现bean的autowire属性被设置成byname,之后容器试图匹配、装配和该bean的属性具有相同名字的bean。

byType::通过参数类型自动装配,Spring容器在配置文件中发现bean的autowire属性被设置成byType,之后容器试图匹配、装配和该bean的属性具有相同类型的bean。如果有多个bean符合条件,则抛出错误。

constructor:这个方式类似于byType, 但是要提供给构造器参数,如果没有确定的带参数的构造器参数类型,将会抛出异常。

autodetect:首先尝试使用constructor来自动装配,如果无法工作,则使用byType方式。

33.自动装配有哪些局限性 ?

自动装配的局限性是:

重写: 你仍需用 <constructor-arg>和 <property> 配置来定义依赖,意味着总要重写自动装配。

基本数据类型:你不能自动装配简单的属性,如基本数据类型,String字符串,和类。

模糊特性:自动装配不如显式装配精确,如果有可能,建议使用显式装配。

34. 你可以在Spring中注入一个null 和一个空字符串吗?

可以。

Spring注解

35. 什么是基于Java的Spring注解配置? 给一些注解的例子。

基于Java的配置,允许你在少量的Java注解的帮助下,进行你的大部分Spring配置而非通过XML文件。

以@Configuration 注解为例,它用来标记类可以当做一个bean的定义,被Spring IOC容器使用。另一个例子是@Bean注解,它表示此方法将要返回一个对象,作为一个bean注册进Spring应用上下文。

36. 什么是基于注解的容器配置?

相对于XML文件,注解型的配置依赖于通过字节码元数据装配组件,而非尖括号的声明。

开发者通过在相应的类,方法或属性上使用注解的方式,直接组件类中进行配置,而不是使用xml表述bean的装配关系。

37. 怎样开启注解装配?

注解装配在默认情况下是不开启的,为了使用注解装配,我们必须在Spring配置文件中配置 <context:annotation-config/>元素。

38. @Required 注解

这个注解表明bean的属性必须在配置的时候设置,通过一个bean定义的显式的属性值或通过自动装配,若@Required注解的bean属性未被设置,容器将抛出BeanInitializationException。

39. @Autowired 注解

@Autowired 注解提供了更细粒度的控制,包括在何处以及如何完成自动装配。它的用法和@Required一样,修饰setter方法、构造器、属性或者具有任意名称和/或多个参数的PN方法。

40. @Qualifier 注解

当有多个相同类型的bean却只有一个需要自动装配时,将@Qualifier 注解和@Autowire 注解结合使用以消除这种混淆,指定需要装配的确切的bean。

Spring数据访问

41.在Spring框架中如何更有效地使用JDBC?

使用SpringJDBC 框架,资源管理和错误处理的代价都会被减轻。所以开发者只需写statements 和 queries从数据存取数据,JDBC也可以在Spring框架提供的模板类的帮助下更有效地被使用,这个模板叫JdbcTemplate (例子见这里here)

42. JdbcTemplate

JdbcTemplate 类提供了很多便利的方法解决诸如把数据库数据转变成基本数据类型或对象,执行写好的或可调用的数据库操作语句,提供自定义的数据错误处理。

43. Spring对DAO的支持

Spring对数据访问对象(DAO)的支持旨在简化它和数据访问技术如JDBC,Hibernate or JDO 结合使用。这使我们可以方便切换持久层。编码时也不用担心会捕获每种技术特有的异常。

44. 使用Spring通过什么方式访问Hibernate?

在Spring中有两种方式访问Hibernate:

控制反转 Hibernate Template和 Callback。

继承 HibernateDAOSupport提供一个AOP 拦截器。

45. Spring支持的ORM

Spring支持以下ORM:

Hibernate

iBatis

JPA (Java Persistence API)

TopLink

JDO (Java Data Objects)

OJB

46.如何通过HibernateDaoSupport将Spring和Hibernate结合起来?

用Spring的 SessionFactory 调用 LocalSessionFactory。集成过程分三步:

配置the Hibernate SessionFactory。

继承HibernateDaoSupport实现一个DAO。

在AOP支持的事务中装配。

47. Spring支持的事务管理类型

Spring支持两种类型的事务管理:

编程式事务管理:这意味你通过编程的方式管理事务,给你带来极大的灵活性,但是难维护。

声明式事务管理:这意味着你可以将业务代码和事务管理分离,你只需用注解和XML配置来管理事务。

48. Spring框架的事务管理有哪些优点?

它为不同的事务API 如 JTA,JDBC,Hibernate,JPA 和JDO,提供一个不变的编程模式。

它为编程式事务管理提供了一套简单的API而不是一些复杂的事务API如

它支持声明式事务管理。

它和Spring各种数据访问抽象层很好得集成。

49. 你更倾向用那种事务管理类型?

大多数Spring框架的用户选择声明式事务管理,因为它对应用代码的影响最小,因此更符合一个无侵入的轻量级容器的思想。声明式事务管理要优于编程式事务管理,虽然比编程式事务管理(这种方式允许你通过代码控制事务)少了一点灵活性。

Spring面向切面编程(AOP)

50. 解释AOP

面向切面的编程,或AOP, 是一种编程技术,允许程序模块化横向切割关注点,或横切典型的责任划分,如日志和事务管理。

51. Aspect 切面

AOP核心就是切面,它将多个类的通用行为封装成可重用的模块,该模块含有一组API提供横切功能。比如,一个日志模块可以被称作日志的AOP切面。根据需求的不同,一个应用程序可以有若干切面。在Spring AOP中,切面通过带有@Aspect注解的类实现。

52. 在Spring AOP 中,关注点和横切关注的区别是什么?

关注点是应用中一个模块的行为,一个关注点可能会被定义成一个我们想实现的一个功能。

横切关注点是一个关注点,此关注点是整个应用都会使用的功能,并影响整个应用,比如日志,安全和数据传输,几乎应用的每个模块都需要的功能。因此这些都属于横切关注点。

53. 连接点

连接点代表一个应用程序的某个位置,在这个位置我们可以插入一个AOP切面,它实际上是个应用程序执行Spring AOP的位置。

54. 通知

通知是个在方法执行前或执行后要做的动作,实际上是程序执行时要通过SpringAOP框架触发的代码段。

Spring切面可以应用五种类型的通知:

before:前置通知,在一个方法执行前被调用。

after: 在方法执行之后调用的通知,无论方法执行是否成功。

after-returning: 仅当方法成功完成后执行的通知。

after-throwing: 在方法抛出异常退出时执行的通知。

around: 在方法执行之前和之后调用的通知。

55. 切点

切入点是一个或一组连接点,通知将在这些位置执行。可以通过表达式或匹配的方式指明切入点。

56. 什么是引入?

引入允许我们在已存在的类中增加新的方法和属性。

57. 什么是目标对象?

被一个或者多个切面所通知的对象。它通常是一个代理对象。也指被通知(advised)对象。

58. 什么是代理?

代理是通知目标对象后创建的对象。从客户端的角度看,代理对象和目标对象是一样的。

59. 有几种不同类型的自动代理?

BeanNameAutoProxyCreator

DefaultAdvisorAutoProxyCreator

Metadata autoproxying

60. 什么是织入。什么是织入应用的不同点?

织入是将切面和到其他应用类型或对象连接或创建一个被通知对象的过程。

织入可以在编译时,加载时,或运行时完成。

61. 解释基于XML Schema方式的切面实现。

在这种情况下,切面由常规类以及基于XML的配置实现。

62. 解释基于注解的切面实现

在这种情况下(基于@AspectJ的实现),涉及到的切面声明的风格与带有java5标注的普通java类一致。

Spring 的MVC

63. 什么是Spring的MVC框架?

Spring 配备构建Web 应用的全功能MVC框架。Spring可以很便捷地和其他MVC框架集成,如Struts,Spring 的MVC框架用控制反转把业务对象和控制逻辑清晰地隔离。它也允许以声明的方式把请求参数和业务对象绑定。

64. DispatcherServlet

Spring的MVC框架是围绕DispatcherServlet来设计的,它用来处理所有的HTTP请求和响应。

65. WebApplicationContext

WebApplicationContext 继承了ApplicationContext 并增加了一些WEB应用必备的特有功能,它不同于一般的ApplicationContext ,因为它能处理主题,并找到被关联的servlet。

66. 什么是Spring MVC框架的控制器?

控制器提供一个访问应用程序的行为,此行为通常通过服务接口实现。控制器解析用户输入并将其转换为一个由视图呈现给用户的模型。Spring用一个非常抽象的方式实现了一个控制层,允许用户创建多种用途的控制器。

67. @Controller 注解

该注解表明该类扮演控制器的角色,Spring不需要你继承任何其他控制器基类或引用Servlet API。

68. @RequestMapping 注解

该注解是用来映射一个URL到一个类或一个特定的方处理法上。

作者:吴盐以兑
链接:https://www.zhihu.com/question/50800464/answer/122921043
来源:知乎
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

第一步【0,1, ...rest】: ES6/ ES2015
现在学JS,ES6是必备啦。
第二步【99乘法表】: TypeScript
也可以从这一步开始,但有第一步这个会理解的更快。同时,这一步我觉得重的要的还是OO思想,原来JS太开放,各种坑不利于软件工程化。
第三步【方程式】:Angular 2 Basics, Material Design( Lite ) 等
一些基础的概念,如Component, Module, Router, Pipe 等
第四步【排列组合】:Javascript Reactive Programming
因为Angular 2 全面拥抱 RxJs, 所以这个我觉得很重要,主是是Observable的思想。
第五步【数列极限】:Redux 概念
不需要学 React + Redux , 主要是概念和原理。
第六步【几何学】:相关命令工具(wepack, Angular-Cli)
webpack以前做前端的已懂的略过。
调试及打包。最新ng tools已经内置Wepack并作为默认打包工具。(完美)
Webpack 套路与 gulp grunt不一样。
第七步 【行列式】:测试
哎,想当年哪能想到前端也要写单元测试。
第八步【高考啦】:Angular2, RxJS, and ngrx/store 开始做个像样的项目!
!!! 此处才能有有高潮,官方教程没有!!!

(级别只是类比,不准确可以建议修改,但求别喷