Struts2即支持属性驱动,也支持模型驱动

属性驱动:在Action中提供与表单字段一一对应的属性,然后一一set赋值

模型驱动:使得表单字段都自动被set到一个JavaBean中,类似于Struts1.XActionForm

采用属性驱动的方式时,是由每个属性来承载表单的字段值,运转在MVC流程里面

采用模型驱动的方式时,是由模型对象来承载所有的属性值,运转在MVC流程里面

若使用模型驱动方式的话,就必须单独提供一个JavaBean

可能与持久层打交道的JavaBean不太一样,因此很可能要对每个对象提供两个JavaBean

比如Struts1的ActionForm,一般来说绝对不会把ActionForm作为JavaBean跟持久层交互

ActionForm仅仅是联系Web和Action的一个桥梁,因此推荐使用属性驱动接收表单字段


com.opensymphony.xwork2.ModelDriven接口源代码中有一段很重要的说明,现抄录如下

ModelDriven Actions provide a model object to be pushed onto the ValueStack in addition

to the Action itself,allowing a FormBean type approach like Struts

翻译:模型驱动的Action。将模型对象以及Action对象都放到ValueStack里面

         允许像Struts一样的FormBean方式

也即:一个Action要想成为模型驱动的话,就必须实现ModelDriven接口

         而我们之前所一直继承的ActionSupport类并没有实现ModelDriven接口

以下是采用模型驱动的Action代码示例

  1. public class ModelDrivenAction extends ActionSupport implements ModelDriven<User> {
  2. private User user = new User();
  3. public User getModel() {
  4. System.out.println(user);
  5. return user;
  6. }
  7. public String execute() throws Exception {
  8. System.out.println("Username is :" + user.getUsername());
  9. System.out.println("Password is :" + user.getPassword());
  10. return SUCCESS;
  11. }
  12. }

ModelDrivenAction类的执行流程是:首先调用getModel()方法得到User对象

接着根据JavaBean的原则将客户端传过来的属性,一个一个的set到User对象的属性中

将属性全部set完之后,再执行execute()方法。对于模型驱动,只要了解这些就足够了


模型驱动的底层实现机制

这里用到了defaultStack拦截器栈中的modelDriven拦截器

它对应com.opensymphony.xwork2.interceptor.ModelDrivenInterceptor类,其API描述如下

public class ModelDrivenInterceptor extends AbstractInterceptor

Watches for ModelDriven actions and adds the action`s model on to the value stack.

翻译:观察模型驱动的Action,并将这个Action的模型【这里指User对象】放到值栈中

Note:The ModelDrivenInterceptor must come before the both StaticParametersInterceptor

and ParametersInterceptor if you want the parameters to be applied to the model.

翻译:若希望将表单提交过来的参数应用到模型里面

         那么ModelDrivenInterceptor拦截器就必须位于StaticParametersInterceptorParametersInterceptor拦截器前面

实际上struts-default.xml已完成这个工作了。可以在defaultStack拦截器栈中查看三者位置

所以对于采用模型驱动的方式的话,在struts.xml中只需要指定模型驱动的类就可以了

其它的都不需要我们手工修改


ModelDrivenInterceptor的部分源代码如下所示

  1. public class ModelDrivenInterceptor extends AbstractInterceptor{  
  2.     public String intercept(ActionInvocation invocation) throws Exception{  
  3.         Object action = invocation.getAction();  
  4.         // 这个action就是当前拦截器准备拦截的Action对象  
  5.         // 我们这里的action就是ModelDrivenAction,它实现了ModelDriven<User>接口  
  6.         // 根据多态性,可认为子类对象就是父类的一个实例,故action属于ModelDriven的实例  
  7.         if(action instanceof ModelDriven){  
  8.             // 将action强制转换为ModelDriven类型,获得ModelDriven的实例  
  9.             ModelDriven modelDriven = (ModelDriven)action;  
  10.             ValueStack stack = invocation.getStack();  
  11.             // 调用getModel(),此时ModelDrivenAction中的getModel()才会被执行,获得User对象  
  12.             Object model = modelDriven.getModel();  
  13.                 if(model != null){  
  14.                     // 获得User对象之后,就把它推入【压入】到值栈中,供后面调用  
  15.                     stack.push(model);  
  16.                 }  
  17.                 //以下略..  
04-15 02:39