Spring Web Flow 是Spring MVC 的扩展,它支持开发基于流程的应用程序。它将流程的定义与实现流程行为的类和视图分离开来。
在Spring MVC上下文定义,添加的命名空间:
xmlns:flow="http://www.springframework.org/schema/webflow-config"
xsi:schemaLocation="
http://www.springframework.org/schema/webflow-config
http://www.springframework.org/schema/webflow=config/spring-webflow-config-2.3.xsd
完整配置:
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:flow="http://www.springframework.org/schema/webflow-config"
xsi:schemaLocation=" http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans-2.5.xsd
http://www.springframework.org/schema/webflow-config
http://www.springframework.org/schema/webflow-config/spring-webflow-config-2.0.xsd">
<flow:flow-executor id="flowExecutor"/>
<flow:flow-registry id="flowRegistry" base-path="/WEB-INF/flows">
<flow:flow-location-pattern value="*-flow.xml"/>
<flow:flow-location path="/WEB-INF/flows/springpizza.xml" id="pizza"/>
flow:flow-registry>
<bean class="org.springframework.webflow.mvc.servlet.FlowHandlerMapping">
<property name="flowRegistry" ref="flowRegistry"/>
bean>
<bean class="org.springframework.webflow.mvc.servlet.FlowHandlerAdapter">
<property name="flowExecutor" ref="flowExecutor"/>
bean>
beans>
流程执行器:会为用户创建并启动一个流程执行实例。当流程暂停的时候(如为用户展示视图时),流程执行器会在用户执行操作后恢复流程。
流程注册表:加载流程定义文件并让流程执行器能够使用它们。
个人认为,Spring WebFlow的实现的核心就是基于其实现了HandlerMapping和HandlerAdapter。
流程是由三个主要元素定义的:状态、转移和流程数据。状态是流程中事件发生的地点。转移用来连接状态。流程数据是流程的当前状况。
Spring Web Flow定义了五种不同类型的状态:
状态类型 | 它是用来做什么的 |
---|---|
行为(Action) | 行为状态时流程逻辑发生的地方 |
决策(Decision) | 决策状态将流程纷飞两个方向,他是基于流程数据的评估结果来确定流程方向 |
结束(end) | 结束状态时流程的最后一站,一旦进入结束状态,流程就会终止 |
子流程(Subflo) | 子流程状态会在当前正在运行的流程上下文中启动一个新的流程 |
视图(view) | 视图状态会暂停流程并邀请用户参与流程 |
视图状态:
<view-state id="welcome" view="greeting" model="flowScope.paymentDetail"/>
id属性标示这个状态,除此之外,如果没有定义view属性,则将自动默认view属性为id属性。model属性表明逻辑视图名greeting所表示的视图中的表单将绑定流程作用域中的paymentDetail对象。
行为状态:
<action-state id="saveOrder">
<evaluate expression="pizzaFlowActions.saveOrder(order)" />
<transition to="thankyou" />
action-state>
上面
给出了行为状态要做的事情。expression属性指定了进入状态时要评估的表达式,这里使用的SpEL表达式,表明将会找到ID为pizzaFlowActions的bean并调用其saveOrder()方法。
决策状态:
<decision-state id="checkDeliverArea">
<if test="pizzaFlowActions.checkDeliverArea(customer.zipCode)"
than="addCustomer"
else="deliverWarning" />
</decision-state>
子流程状态:
id="order" subflow="pizza/order">
name="order" value="order" />
on="orderCreated" to="payment" />
在这里,元素用于传递订单对象作为子流程的输入,如果子流程结束的
状态ID为orderCreated,那么流畅将会转移到名为payment的状态。
结束状态:
<end-state id="customerReady" />
当到达
状态,流程会结束。接下来会发生什么取决于几个因素:
处继续执行,
的ID将会用作事件触发从
开始的转移。
设置了view属性,指定的视图将会被渲染。视图可以是相对于流程路径的视图模板,如果添加“externalRedirect:”前缀的话,将会重定向到流程外部的页面,如果添加“flowRedirect:”将重定向到另外一个流程中。流程中除结束状态之外的每个状态,至少都需要一个转移,这样就能够知道一旦这个状态完成时流程要去向哪里。状态可以有多个转移,分别对应于当前状态结束时可以执行的不同的路径。
to="customerReady" on="phoneEntered"/>
to属性指定流程的下一个状态,必须。on属性指定触发转移的事件,非必须。
还可以使用on-exception属性指定抛出异常时进入另一个状态。
on-exception=
"com.springination.pizza.service.CustomNotFoundException" to="registrationForm" />
全局转移(流程中的所有状态都会默认拥有这个转移):
<globl-transitions>
<transition on="cancel" to="endState" />
</global-transitions>
声明变量:
使用
<var name="customer" class="com.xxx.Customer" />
使用set
<set name="flowScope.pizza"
value="new com.xx.Pizza()" />
作为行为状态的一部分或者作为视图状态的入口,可以使用evaluate元素来创建变量:
<evaluate result="viewScope.toppingsList"
expression="T(com.xxx.Topping).asList()" />
这里计算了一个表达式SpELl并将结果放到了名为toppingList的变量中,这个变量是视图作用域的。
set和evaluate元素很类似,都是讲变量设置为表达式计算的结果。
Spring WebFlow的作用域:
范围 | 生命作用域和可见性 |
---|---|
Conversation | 最高层级的流程开始时创建,在最高层级的流程结束时销毁。被最高层级的流程和其所有的子流程所共享 |
Flow | 当流程开始时创建,在流程结束时销毁。只有在创建它的流程中是可见的 |
Request | 当一个请求进入流程时创建,在流程返回时销毁 |
Flash | 当流程开始时创建,在流程结束时销毁。在视图状态渲染后,它也会被清除 |
View | 当进入视图状态时创建,当这个状态退出时销毁,只是在视图状态内是可见的 |
当时用元素声明变量时,变量始终是流程作用域,也就是在定义变量的流程内有效。当使用
或
的时候,作用域通过name或result属性的前缀指定。
<set name="flowScop.theAnswer" value="32">
按照书上做的一个订购披萨的流程Demo,Demo下载链接:http://download.csdn.net/detail/csdn_xuexiaoqiang/9798188,搭建好后流程的访问地址是“localhost:8080/spizza/pizza”。
2017-4-24 21:50:42补充集成
只需要配置两个DispatcherServlet就可以了。
下面是web.xml
<web-app version="2.5"
xmlns="http://java.sun.com/xml/ns/javaee"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance
http://www.springmodules.org/schema/cache/springmodules-cache.xsd
http://www.springmodules.org/schema/cache/springmodules-ehcache.xsd"
xsi:schemaLocation="http://java.sun.com/xml/ns/javaee
http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd">
<context-param>
<param-name>contextConfigLocationparam-name>
<param-value>classpath:applicationContext.xmlparam-value>
context-param>
<listener>
<listener-class>org.springframework.web.context.ContextLoaderListenerlistener-class>
listener>
<servlet>
<servlet-name>springservlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServletservlet-class>
<init-param>
<param-name>contextConfigLocationparam-name>
<param-value>classpath:dispatcherServlet-context.xmlparam-value>
init-param>
<load-on-startup>1load-on-startup>
servlet>
<servlet-mapping>
<servlet-name>springservlet-name>
<url-pattern>*.dourl-pattern>
servlet-mapping>
<servlet>
<servlet-name>pizzaServletservlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServletservlet-class>
<init-param>
<param-name>contextConfigLocationparam-name>
<param-value>classpath:pizaarServlet-context.xmlparam-value>
init-param>
<load-on-startup>1load-on-startup>
servlet>
<servlet-mapping>
<servlet-name>pizzaServletservlet-name>
<url-pattern>/flow/*url-pattern>
servlet-mapping>
web-app>
配置文件pizaarServlet-context.xml
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:flow="http://www.springframework.org/schema/webflow-config"
xsi:schemaLocation=" http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans-2.5.xsd
http://www.springframework.org/schema/webflow-config
http://www.springframework.org/schema/webflow-config/spring-webflow-config-2.0.xsd">
<bean id="viewResolver" class="org.springframework.web.servlet.view.InternalResourceViewResolver">
<property name="prefix" value="/WEB-INF/"/>
<property name="suffix" value=".jsp"/>
bean>
<flow:flow-executor id="flowExecutor"/>
<flow:flow-registry id="flowRegistry">
<flow:flow-location path="/WEB-INF/pizza-flow.xml" id="pizza"/>
<flow:flow-location path="/WEB-INF/customer-flow.xml" id="customer"/>
<flow:flow-location path="/WEB-INF/order-flow.xml" id="order"/>
<flow:flow-location path="/WEB-INF/payment-flow.xml" id="payment"/>
flow:flow-registry>
<bean id="handlerMapping" class="org.springframework.webflow.mvc.servlet.FlowHandlerMapping">
<property name="flowRegistry" ref="flowRegistry"/>
bean>
<bean id="handlerAdapter" class="org.springframework.webflow.mvc.servlet.FlowHandlerAdapter">
<property name="flowExecutor" ref="flowExecutor"/>
bean>
beans>
配置文件dispatcherServlet-context.xml
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:aop="http://www.springframework.org/schema/aop"
xmlns:tx="http://www.springframework.org/schema/tx" xmlns:context="http://www.springframework.org/schema/context"
xsi:schemaLocation="
http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.0.xsd
http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-3.0.xsd
http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-3.0.xsd
http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-3.0.xsd">
<bean id="viewResolver" class="org.springframework.web.servlet.view.InternalResourceViewResolver">
<property name="prefix" value="/WEB-INF/"/>
<property name="suffix" value=".jsp"/>
bean>
<context:component-scan base-package="com.pizza.action"/>
beans>
这样就可以了。通过localhost:8080/spizza/flow/pizza
这个链接就可以访问我们定义的披萨流程。
通过localhost:8080/spizza/index.do
就可以访问下面这个controller:
@Controller
public class Index {
@RequestMapping("/index.do")
public void test(){
System.out.println("MVC测试成功");
}
}