Spring学习笔记之Spring Web Flow

Spring Web Flow 是Spring MVC 的扩展,它支持开发基于流程的应用程序。它将流程的定义与实现流程行为的类和视图分离开来。

1.配置Web Flow

在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。

2.流程组件

流程是由三个主要元素定义的:状态、转移和流程数据。状态是流程中事件发生的地点。转移用来连接状态。流程数据是流程的当前状况。

2.1 状态

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:”将重定向到另外一个流程中。
  • 如果结束的流程不是子流程,也没有指定view属性,那这个流程只是会结束而已。浏览器最后将会加载流程的基本URL地址,当前已没有活动的流程,所以会开始一个新的流程实例。

2.2转移

流程中除结束状态之外的每个状态,至少都需要一个转移,这样就能够知道一旦这个状态完成时流程要去向哪里。状态可以有多个转移,分别对应于当前状态结束时可以执行的不同的路径。

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>

2.3流程数据

声明变量:
使用

<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">

3.案例

按照书上做的一个订购披萨的流程Demo,Demo下载链接:http://download.csdn.net/detail/csdn_xuexiaoqiang/9798188,搭建好后流程的访问地址是“localhost:8080/spizza/pizza”。


2017-4-24 21:50:42补充集成

4.Spring MVC和Spring Web Flow集成

只需要配置两个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测试成功");
    }
}

你可能感兴趣的:(Spring学习笔记)