`
_与狼共舞
  • 浏览: 62272 次
  • 性别: Icon_minigender_1
  • 来自: 成都
社区版块
存档分类
最新评论

Struts2中的ModelDriven机制及运用

    博客分类:
  • Java
阅读更多

所谓ModelDriven,意思是直接把实体类当成页面数据的收集对象 。

 

比如,有实体类User如下:

package cn.com.leadfar.struts2.actions;

 

public class User {

    private int id;

    private String username;

    private String password;

    private int age;

    private String address;

    public String getUsername() {

       return username;

    }

    public void setUsername(String username) {

       this.username = username;

    }

    public String getPassword() {

       return password;

    }

    public void setPassword(String password) {

       this.password = password;

    }

    public int getAge() {

       return age;

    }

    public void setAge(int age) {

       this.age = age;

    }

    public String getAddress() {

       return address;

    }

    public void setAddress(String address) {

       this.address = address;

    }

    public int getId() {

       return id;

    }

    public void setId(int id) {

       this.id = id;

    }

}

 

假如要写一个Action,用来添加User

第一种做法是直接在Action中定义所有需要的属性,然后在JSP中直接用属性名称来提交数据:

UserAction:

 

public class UserAction {

    private int id;

    private String username;

    private String password;

    private int age;

    private String address;

 

    public String add(){

      

       User user = new User();

       user.setId(id);

       user.setUsername(username);

       user.setPassword(password);

       user.setAge(age);

       user.setAddress(address);

      

       new UserManager().addUser(user);

      

       return "success";

    }

   

    public int getId() {

       return id;

    }

    public void setId(int id) {

       this.id = id;

    }

    public String getUsername() {

       return username;

    }

    public void setUsername(String username) {

       this.username = username;

    }

    public String getPassword() {

       return password;

    }

    public void setPassword(String password) {

       this.password = password;

    }

    public int getAge() {

       return age;

    }

    public void setAge(int age) {

       this.age = age;

    }

    public String getAddress() {

       return address;

    }

    public void setAddress(String address) {

       this.address = address;

    } 

}

 

 

add_input.jsp:

 

     <form action="test/user.action" method="post">

        <input type="hidden" name="method:add">

        username:<input type="text" name="username"> <br/>

        password:<input type="text" name="password"> <br/>

        age:<input type="text" name="age"> <br/>

        address:<input type="text" name="address"> <br/>

        <input type="submit" name="submit" value="添加用户">

     </form> <br/>

 

上述做法不好之处是:如果实体类的属性非常多,那么Action中也要定义相同的属性

 

第二种做法是将User对象定义到UserAction中,然后在JSP中通过user属性来给user赋值:

UserAction:

 

public class UserAction {

   

    private User user;

   

    public String add(){

 

       new UserManager().addUser(user);

      

       return "success";

    }

 

    public User getUser() {

       return user;

    }

 

    public void setUser(User user) {

       this.user = user;

    } 

}

 

 

add_input.jsp:

 

     <form action="test/user.action" method="post">

        <input type="hidden" name="method:add">

        username:<input type="text" name="user.username"> <br/>

        password:<input type="text" name="user.password"> <br/>

        age:<input type="text" name="user.age"> <br/>

        address:<input type="text" name="user.address"> <br/>

        <input type="submit" name="submit" value="添加用户">

     </form> <br/>

 

这种做法不好的地方是:JSP页面上表单域中的命名变得太长。

 

第三种做法是利用ModelDriven机制,让UserAction实现一个ModelDriven接口,同时实现接口中的方法:getModel()如下所示:

public class UserAction implements ModelDriven{

   

    private User user;

   

    @Override

    public Object getModel() {

       if(user == null){

           user = new User();

       }

       return user;

    }

 

    public String add(){

 

       new UserManager().addUser(user);

      

       return "success";

    }

 

    public User getUser() {

       return user;

    }

 

    public void setUser(User user) {

       this.user = user;

    }

}

JSP的代码如下:

     <form action="test/user.action" method="post">

        <input type="hidden" name="method:add">

        username:<input type="text" name="username"> <br/>

        password:<input type="text" name="password"> <br/>

        age:<input type="text" name="age"> <br/>

        <input type="submit" name="submit" value="添加用户">

     </form> <br/>

 

可见,第三种做法是比较好的,ActionJSP写起来都比较简单。

 

 

 

ModelDriven机制?

ModelDriven背后的机制就是ValueStack界面通过:username/age/address这样的名称,就能够被直接赋值给user对象,这证明user对象正是ValueStack中的一个root对象!

 

那么,为什么user对象会在ValueStack中呢?它是什么时候被压入ValueStack的呢?答案是:ModelDrivenInterceptor。ModelDrivenInterceptor是缺省的拦截器链的一部分,当一个请求经过ModelDrivenInterceptor的时候,在这个拦截器中,会判断当前要调用的Action对象是否实现了ModelDriven接口,如果实现了这个接口,则调用getModel()方法,并把返回值(本例是返回user对象)压入ValueStack

请看ModelDrivenInterceptor的代码:

 

public class ModelDrivenInterceptor extends AbstractInterceptor {

 

    protected boolean refreshModelBeforeResult = false;

 

    public void setRefreshModelBeforeResult(boolean val) {

        this.refreshModelBeforeResult = val;

    }

 

    @Override

    public String intercept(ActionInvocation invocation) throws Exception {

        Object action = invocation.getAction();

 

        if (action instanceof ModelDriven) {

            ModelDriven modelDriven = (ModelDriven) action;

            ValueStack stack = invocation.getStack();

            Object model = modelDriven.getModel();

            if (model !=  null) {

              stack.push(model);

            }

            if (refreshModelBeforeResult) {

                invocation.addPreResultListener(new RefreshModelBeforeResult(modelDriven, model));

            }

        }

        return invocation.invoke();

    }

 

ModelDrivenInterceptor中,即可以看到model对象被压入ValueStack中!

 

其中的refreshModelBeforeResult是为了接下来描述的一个问题而提供的解决方法。

 

理解常见的陷阱及其解决方法

假设我们要更新一个实体对象,那么第一步首先是打开更新界面,请看下述模拟打开更新界面的代码:

 

public class UserAction implements ModelDriven{

   

    private User user;

   

    @Override

    public Object getModel() {

       if(user == null){

           user = new User();

           //user.setUsername("这是原来的User对象");

       }

       return user;

    }

   

    public String updateInput(){

      

       //根据ID,查询数据库,得到User对象

       user = new UserManager().findUserById(user.getId());

       return "update_input";

    }

 

 

上述代码中,new UserManager().findUserById(user.getId());这一行,将从数据库中查询相应的记录,同时转换为User对象返回。而return update_input;将转向更新显示页面。

 

更新页面如下:

 

     <form action="test/user.action" method="post">

        <input type="hidden" name="method:update">

        id:<input type="text" name="id" value="<s:property value="id"/>"> <br/>

        username:<input type="text" name="username" value="<s:property value="username"/>"><br/>

        password:<input type="text" name="password" value="<s:property value="password"/>"><br/>

        age:<input type="text" name="age" value="<s:property value="age"/>"> <br/>

        address:<input type="text" name="address" value="<s:property value="address"/>"><br/>

        <input type="submit" name="submit" value="更新用户">

     </form> <br/>

 

 

上述代码运行起来之后,你在更新界面上将看不到数据(id属性有值,其它属性无显示)。关键的原因是在执行到updateInput之前,user对象(在getMode()方法中创建的对象)被压到ValueStack中,这时候,UserActionValueStack都指向同一个user对象;但紧接着,UserAction中的user被一个新的user对象覆盖,这时候,UserActionValueStack不再指向同一个user对象!ValueStack中是旧的user对象,而UserAction中是新的user对象!我们在JSP中,直接通过username/address等直接访问,当然是要访问ValueStack中的旧user对象,所以它们的属性都是空的(id属性除外)

 

理解上述问题很重要,当你理解了问题,那么问题的解决方法就可以有很多了:

比如,你可以把新对象的属性拷贝到旧对象上;比如,你可以先把旧对象从ValueStack中移除,然后再把新对象压入ValueStack等……

 

在最新的struts2版本中,ModelDrivenInterceptor提供了一个配置参数:refreshModelBeforeResult,只要将它定义为true,上述问题就被解决了!struts2的解决方案就是:先把旧的model对象从ValueStack中移除,然后再把新的model对象压入ValueStack

分享到:
评论

相关推荐

    Struts2_ModelDriven接收参数(项目案例及视频)

    在Struts2中,ModelDriven接口是一个非常实用的功能,它可以帮助我们更方便地处理Action类中的模型数据。这个项目案例和视频教程主要针对初学者,旨在深入理解如何使用Struts2的ModelDriven机制来接收和传递参数。 ...

    Struts2接收参数ModelDriven

    总之,Struts2的ModelDriven机制是其框架中一个强大的功能,可以帮助开发者更好地管理模型数据,提高代码的简洁性和可维护性。通过理解并熟练运用这一机制,可以提升开发效率,同时也要注意评估其可能带来的性能影响...

    struts2和jsp的运用

    在Struts2中,我们可以创建一个Action类,定义对应的方法(如addAction、deleteAction、updateAction和listAction),然后配置struts.xml文件以指定这些方法与URL的映射。对于每个方法,我们可以通过ModelDriven接口...

    Struts2视频教程

    ### Struts2核心知识点解析 #### 一、Struts2框架概述 - **定义与特点**:Struts2是一款基于MVC(Model-View-Controller)设计模式的...希望本教程能帮助大家更好地理解和运用Struts2,为自己的项目带来更多的价值。

    ModelDriven

    在Struts2中,ModelDriven接口主要用于处理Action类中的模型数据,使得开发者无需在Action类中定义所有可能的业务对象属性,而是通过实现该接口将模型直接绑定到请求参数上。 【拦截器(Interceptor)】是Struts2...

    struts2的核心jar包

    - Struts2中的Action是业务逻辑的载体,它是用户请求处理的主要对象。Action类可以继承自`org.struts2.interceptor.ActionSupport`,并实现相应的业务方法。 - 拦截器是Struts2处理请求的重要机制,它们在Action...

    struts2 JAR包合集.zip

    7. **异常处理**:Struts2提供了全局的异常处理机制,可以统一处理程序中的异常,避免了在每个Action中单独处理异常的繁琐。 8. **国际化与本地化**:Struts2支持多语言环境,通过资源文件可以实现内容的国际化和...

    <<深入浅出Struts 2>>中文版 下载

    8. **异常处理**:学习如何在Struts 2中优雅地处理异常,包括全局和局部的异常映射。 9. **整合其他技术**:了解Struts 2与其他技术的集成,如Spring、Hibernate、MyBatis等,构建完整的Java EE应用。 10. **最佳...

    struts2 教学管理系统(含数据库)

    9. **异常处理**:Struts2提供了一套完善的异常处理机制,可以捕获并优雅地处理运行时异常,保持系统的稳定性和用户体验。 10. **国际化(i18n)**:对于多语言支持,Struts2有内置的支持,通过资源文件实现界面...

    struts2中的action.doc

    在Struts2中,Action类是核心组件,它负责处理用户的请求并协调应用逻辑。本文将深入探讨Struts2 Action中的数据处理机制,特别是关于数据来源和页面数据与Action的对应方式。 首先,Action中的数据主要来源于用户...

    【动力节点】王勇老师Java视频教程_Java框架视频之Struts2

    这是一套面向已掌握Java基础和Web知识的学习者设计的课程,旨在帮助学员深入理解并熟练运用Struts2框架,从而提升其在实际项目中的开发能力。 Struts2是Java EE领域中广泛使用的MVC框架之一,它的出现极大地简化了...

    struts2 in action

    在模型层方面,《Struts2 in Action》讲解了如何将业务逻辑集成到Struts2框架中,包括使用ActionSupport基类、ModelDriven接口以及ValueStack等特性。书中还会介绍如何与持久层框架如Hibernate或MyBatis进行集成,...

    struts2学习笔记十(第10讲.Struts2的核心 拦截器)

    在Struts2中,拦截器扮演着至关重要的角色,它们是实现业务逻辑和控制流程分离开来的重要组件。本篇将深入探讨Struts2的核心拦截器以及其工作原理。 在Struts2中,拦截器是一个实现了`Interceptor`接口的类,它通过...

    简单的struts2小程序

    Struts2是一个强大的Java web开发...通过这个小程序,开发者可以了解如何在实际项目中运用Struts2来处理用户请求、执行业务逻辑以及展示结果。这不仅有助于提升Java web开发技能,也为更复杂的应用场景打下了坚实基础。

    struts2权威指南

    7. **OGNL(Object-Graph Navigation Language)**:OGNL是Struts2中的表达式语言,用于在视图层获取和设置值栈中的数据,实现动态表示层的渲染。 8. **拦截器**:拦截器是Struts2的一大特色,它可以定义一系列预定...

    Struts2·0中文教程

    1. **Action和结果类型**:在Struts2中,Action类是业务逻辑处理的主要载体。每个Action对应一个URL,处理特定的用户请求,并通过结果类型(Result)来决定响应的结果页面或跳转路径。 2. **拦截器(Interceptor)*...

    Struts2 in Action

    《Struts2 in Action》是一本深入探讨Struts2框架的专著,旨在帮助开发者全面理解和熟练运用这一流行的Java Web开发框架。Struts2作为Apache软件基金会的一个项目,是MVC(模型-视图-控制器)设计模式在Web应用程序...

    Struts2级联doubleselect

    在Struts2中,级联(Cascading)功能主要用于处理关联数据,例如一个国家对应多个城市,一个城市对应多个区,这种层级关系的数据在用户界面中通常通过级联选择框(dropdown list)来呈现。Doubleselect是Struts2提供...

    Struts2_BBS

    Struts2_BBS 是一个基于Struts2框架构建的论坛(BBS)示例项目,旨在展示如何在实际开发中运用Struts2进行Web应用的搭建。Struts2是一个强大的、开源的MVC(Model-View-Controller)框架,它在Java Web开发中广泛...

Global site tag (gtag.js) - Google Analytics