• 背景:

通过上一章节《Struts(十六):通过CURD来学习Struts流程及ModelDriven的用法》学习了ModelDriven拦截器的用法,上章节中讲到了edit功能。

要修改一个member信息时:

1、首先通过url传入一个id参数:member-list.jsp中

            <td><s:a href="member-view.action?id=%{id}" >view</s:a></td>
<td><s:a href="member-edit.action?id=%{id}" >edit</s:a></td>
<td><s:a href="member-delete.action?id=%{id}" >delete</s:a></td>

2、MemberAction.java的edit函数:

    public String edit() {
// 1、根据id从数据库中,获取id对应的member对象
Member member_ = memberDao.get(this.member.getId()); // 2、把从数据库中获取的属性放入值栈的属性中
this.member.setAge(member_.getAge());
this.member.setName(member_.getName());
this.member.setGender(member_.getGender()); return "edit";
}

3、表单form-edit.jsp:

    <s:debug></s:debug>
<s:form action="member-save.action">
<s:textfield name="id" label="ID"></s:textfield>
<s:textfield name="name" label="Name"></s:textfield>
<s:textfield name="age" label="Age"></s:textfield>
<s:radio list="#{'male':'male','female':'female' }" name="gender" label="Gender"></s:radio>
<s:submit name="submit" label="提交"></s:submit>
</s:form>

流程分析图:

根据上边的代码,调试form-edit.jsp页面当打开s:debug我们确实看到了栈顶对象被填充值了:

4、修改MemberAction.java的edit方法,并调试产看member-edit.jsp debug内容:

    public String edit() {
// // 1、根据id从数据库中,获取id对应的member对象
// Member member_ = memberDao.get(this.member.getId());
//
// // 2、把从数据库中获取的属性放入值栈的属性中
// this.member.setAge(member_.getAge());
// this.member.setName(member_.getName());
// this.member.setGender(member_.getGender());
//
this.member= memberDao.get(this.member.getId()); return "edit";
}

这是为什么呢?

---getModel()返回的对象是存放在栈顶的对象,而修改后的对象是Dao中返回的对象,这两个对象不是同一个对象的原因。

我们是否能通过其他方式来实现这个功能呢?

  • 自动填写了memeber-edit.jsp表单页面改写方案(paramsPrepareParamsStack方案)

方案一:把重新赋值后的this.member对象压入栈顶

    public String edit() {
// // 1、根据id从数据库中,获取id对应的member对象
// Member member_ = memberDao.get(this.member.getId());
//
// // 2、把从数据库中获取的属性放入值栈的属性中
// this.member.setAge(member_.getAge());
// this.member.setName(member_.getName());
// this.member.setGender(member_.getGender()); this.member = memberDao.get(this.member.getId());
ActionContext.getContext().getValueStack().push(this.member); return "edit";
}

不错这样是可以自动填写了memeber-edit.jsp表单页面,但是我们查看s:debug下发现值栈中有两个member对象:

方案二:使用paramsPrepareParams拦截器栈

1、修改struts.xml来实现修改当前项目struts2的拦截器栈

在package标签下添加配置:<default-interceptor-ref name="paramsPrepareParamsStack"></default-interceptor-ref>

<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE struts PUBLIC
"-//Apache Software Foundation//DTD Struts Configuration 2.3//EN"
"http://struts.apache.org/dtds/struts-2.3.dtd"> <struts>
<constant name="struts.ognl.allowStaticMethodAccess" value="true" />
<constant name="struts.devMode" value="false" /> <package name="default" namespace="/" extends="struts-default">
<default-interceptor-ref name="paramsPrepareParamsStack"></default-interceptor-ref>
<action name="member-*" class="com.dx.struts.actions.MemberAction"
method="{1}">
<result name="{1}">/member-{1}.jsp</result>
<result name="delete" type="redirectAction">member-list</result>
<result name="modify" type="redirectAction">member-list</result>
<result name="create" type="redirectAction">member-list</result>
</action>
</package>
<!-- Add packages here -->
</struts>

2、对MemberAction.java进行修改:

1)添加id属性,并实现其set方法,目的是:实现在调用ModelDriven的getModel函数前,通过Params调用该set方法实现对id属性赋值,因此可以在getModel内部使用该属性判定是否页面传递了该参数。

2)修改MemberAction.java自身方法。

修改后代码如下:

/**
* @author Administrator
*
*/
package com.dx.struts.actions; import java.util.Date;
import java.util.List;
import java.util.Map; import org.apache.struts2.interceptor.RequestAware; import com.opensymphony.xwork2.ActionContext;
import com.opensymphony.xwork2.ModelDriven; import com.dx.struts.dao.MemberDao;
import com.dx.struts.entity.Member; /**
* ModelDriven 和Preparable拦截器
*/
public class MemberAction implements RequestAware, ModelDriven<Member> {
private MemberDao memberDao = new MemberDao();
private Member member;
private Long id; // Params拦截器 负责赋值id时,调用了该函数。
public void setId(Long id) {
this.id = id;
} public String list() {
List<Member> members = memberDao.getMembers();
request.put("members", members);
return "list";
} public String view() {
// // 1、获取传入的id:member.getId();
// // 2、根据id获取member对象;
// Member member_ = memberDao.get(this.member.getId());
//
// // 3、赋值栈顶对象的属性:此时栈顶对象为member对象。
// member_.setAge(this.member.getAge());
// member_.setName(this.member.getName());
// member_.setGender(this.member.getGender());
//
return "view";
} public String delete() {
// memberDao.remove(this.member.getId());
memberDao.remove(this.id);
// 返回结果的类型应该为:redirectAction
// 也可以是chain:实际上chain是没有必要的,因为不需要在下一个action中保留啊当前action的状态
// 若使用chain,则到达目标页面后,地址栏显示的依然是删除的那个链接,则刷新时会重复提交。
return "delete";
} public String edit() {
// // 1、根据id从数据库中,获取id对应的member对象
// Member member_ = memberDao.get(this.member.getId());
//
// // 2、把从数据库中获取的属性放入值栈的属性中
// this.member.setAge(member_.getAge());
// this.member.setName(member_.getName());
// this.member.setGender(member_.getGender()); // this.member = memberDao.get(this.member.getId());
// ActionContext.getContext().getValueStack().push(this.member); return "edit";
} public String save() {
//.... return "modify";
} public String create() {
member.setId(new Date().getTime());
memberDao.add(member); return "create";
} private Map<String, Object> request; @Override
public void setRequest(Map<String, Object> request) {
this.request = request;
} @Override
public Member getModel() {
// 如果需要在这个函数调用调用了ModelDriven拦截器栈之前就初始化参数id的拦截器栈的Params拦截器。
// 经过查看DefaultStack拦截器栈中ModelDriven拦截器之前并没有Params拦截器,说明该默认拦截器已经不够用了,
// 但是我们发现在struts2-core.jar下struts-default.xml下除了DefaultStack拦截器栈外还有paramsPrepareParamsStack拦截器栈,
// 而该拦截器栈中恰好在ModelDriven拦截器栈之前和之后都调用了一次Params拦截器。
// 修改struts.xml package标签下添加配置:<default-interceptor-ref name="paramsPrepareParamsStack"></default-interceptor-ref> // 如果没有id请求参数,说明为create
if (this.id == null) {
this.member = new Member();
}
// 如果有id请求参数,说明为edit
else {
this.member = memberDao.get(this.id);
}
return this.member;
}
}
  • 值栈paramsPrepareParamsStack方案总结

不足之处(资源浪费):

1、在执行删除的时候,id不为null,但是getModel方法执行了一次从数据库中空加载;

2、当执行查询信息时,也空执行了一次new Member()对象。

Struts(十七):通过CURD来学习paramsPrepareParams拦截器栈的更多相关文章

  1. Struts(十八):通过CURD来学习PrepareInterceptor拦截器

    PrepareInterceptor拦截器的用法: 1.若Action实现了Preparable接口,则Action方法需实现prepare()方法: 2.PrepareInterceptor拦截器S ...

  2. Struts2学习之拦截器栈

    © 版权声明:本文为博主原创文章,转载请注明出处 拦截器栈: - 从结构上看:拦截器栈相当于多个拦截器的组合 - 从功能上看:拦截器栈也是拦截器 默认拦截器栈: - 在struts-core.jar中 ...

  3. SpringMVC学习 十三 拦截器栈

    拦截器栈:就是有多个拦截器同时拦截相同的控制器(controller)请求,这写拦截器就构成了拦截器栈. 栈的特点是先进后出,在拦截器栈中也是如此,如果先执行了preHandle方法,也就是意味着先进 ...

  4. [原创]java WEB学习笔记66:Struts2 学习之路--Struts的CRUD操作( 查看 / 删除/ 添加) 使用 paramsPrepareParamsStack 重构代码 ,PrepareInterceptor拦截器,paramsPrepareParamsStack 拦截器栈

    本博客的目的:①总结自己的学习过程,相当于学习笔记 ②将自己的经验分享给大家,相互学习,互相交流,不可商用 内容难免出现问题,欢迎指正,交流,探讨,可以留言,也可以通过以下方式联系. 本人互联网技术爱 ...

  5. Struts 2学习笔记——拦截器相关

    一.添加国际化支持 默认的struts-deault.xml文件中已经定义了国际化拦截器,内容如下 <!-定义国际化拦截器--> <interceptor name="i1 ...

  6. 【Java EE 学习 69 上】【struts2】【paramsPrepareParamsStack拦截器栈解决model对象和属性赋值冲突问题】

    昨天有同学问我问题,他告诉我他的Action中的一个属性明明提供了get/set方法,但是在方法中却获取不到表单中传递过来的值.代码如下(简化后的代码) public class UserAction ...

  7. struts2学习笔记--拦截器(Interceptor)和登录权限验证Demo

    理解 Interceptor拦截器类似于我们学过的过滤器,是可以在action执行前后执行的代码.是我们做web开发是经常使用的技术,比如权限控制,日志.我们也可以把多个interceptor连在一起 ...

  8. Struts2基础学习(五)—拦截器

    一.概述 1.初识拦截器      Interceptor 拦截器类似前面学过的过滤器,是可以在action执行前后执行的代码,是我们做Web开发经常用到的技术.比如:权限控制.日志等.我们也可以将多 ...

  9. struts2学习(5)拦截器简介以及例子执行过程

    一.拦截器简介: 二.Struts2预定义拦截器&拦截器栈 在执行action之前和之后,拦截器进行了操作: 比如struts-default.xml中就有很多预定义的拦截器:   拦截器栈: ...

随机推荐

  1. 15.C++-操作符重载

    首先回忆下以前学的函数重载 函数重载 函数重载的本质为相互独立的不同函数 通过函数名和函数参数来确定函数调用 无法直接通过函数名得到重载函数的入口地址 函数重载必然发生在同一个作用域中 类中的函数重载 ...

  2. 001-List,数组,Set,Map属性的映射

    hibernate.cfg.xml: <!DOCTYPE hibernate-configuration PUBLIC "-//Hibernate/Hibernate Configur ...

  3. Docker学习笔记(一)

    什么是Docker? 1.基于Go语言开发的云开源项目,Docker的主要目标是通过对应用组件的 封装,分发,部署,运行等生命周期的管理,达到应用组件级别的 一次封装,到处运行. 2.可以将Docke ...

  4. 部署腾讯云(CentOS6.6版本,jdk1.7+tomcat8+mysql)

    这是从一个大神哪里学到的,用来留下来用以记录 http://blog.csdn.net/qingluoII/article/details/76053736 只是其中有一个地方,我在学习的时候觉得可以 ...

  5. 【highlight.js】页面代码高亮插件

    [highlight.js] 很多博客都支持页面插入各种语言的代码,而这些代码肯定是有高亮设置的.那么在我们自己的页面上如何进行代码高亮设置?有现成的这个highlight.js插件我们可以使用. h ...

  6. C语言描述链表的实现及操作

    一.链表的创建操作 // 操作系统 win 8.1 // 编译环境 Visual Stuido 2017 #include<stdio.h> #include<malloc.h> ...

  7. 将 Shiro 作为应用的权限基础 三:基于注解实现的授权认证过程

    授权即访问控制,它将判断用户在应用程序中对资源是否拥有相应的访问权限. 如,判断一个用户有查看页面的权限,编辑数据的权限,拥有某一按钮的权限等等. 一.用户权限模型 为实现一个较为灵活的用户权限数据模 ...

  8. Java 并发编程实践基础 读书笔记: 第一章 JAVA并发编程实践基础

    1.创建线程的方式: /** * StudySjms * <p> * Created by haozb on 2018/2/28. */ public class ThreadDemo e ...

  9. centos7上安装ffmpeg

    FFmpeg介绍 FFmpeg是一个开源免费跨平台的视频和音频流方案,属于自由软件,采用LGPL或GPL许可证(依据你选择的组件).它提供了录制.转换以及流化音视频的完整解决方案.它包含了非常先进的音 ...

  10. 数据结构——线性表——队列(queue)

    队列也是一种特殊的线性表,它的特点是先入先出(FIFO,即first in first out).它的意思也很直观,想象一下排队买票,先排的人先买(插队是不对的,所以别去想).它也是很常用的数据结构, ...