• 背景:

通过上一章节《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. Python上下文管理器

    在Python中让自己创建的函数.类.对象支持with语句,就实现了上线文管理协议.我们经常使用with open(file, "a+") as f:这样的语句,无需手动调用f.c ...

  2. curl/libcurl获取打开网页平均网速

    CURL: curl -o /dev/null -s -w %{http_code}:%{http_connect}:%{content_type}:%{time_namelookup}:%{time ...

  3. 最小生成数(并查集)Kruskal算法

    并查集:使用并查集可以把每个连通分量看作一个集合,该集合包含连通分量的所有点.这两两连通而具体的连通方式无关紧要,就好比集合中的元素没有先后顺序之分,只有属于和不属于的区别.#define N 100 ...

  4. ReactiveCocoa--RACTuple

    基本信息 例子 [[self rac_signalForSelector:@selector(tableView:didSelectRowAtIndexPath:) fromProtocol:@pro ...

  5. 基于PLC-C#串口通讯,温度检测和转速监控的c#/.Net实现。

    我司为五金加工企业,其中有一条喷涂车间和流水线,客户要求能实时监控炉温温度.流水线速,并设置上下限值,达到上下限时报警. 开始考虑过USB的温度采集器,但是却没有找到带USB的光电开关,并且线路长度受 ...

  6. canvas星空和图形变换

    图形变换. 一.画一片星空 先画一片canvas.width宽canvas.height高的黑色星空,再画200个随机位置,随机大小,随机旋转角度的星星. window.onload=function ...

  7. PO BO VO DTO POJO DAO DO

    PO BO DTO VO 归在一起叫是POJO,简单java对象:DAO 是进行数据库增删改查的类,DO不确定有没有. 重点说下POJO PO 持久对象,数据: BO 业务对象,封装对象.复杂对象 , ...

  8. 功能测试很low?不能升级到高级测试工程师?

    功能测试很low?不能升级到高级测试工程师? 功能测试很low?功能测试很简单?功能测试就是黑盒测试?功能测试没有技术含量?功能测试工资低?只会功能测试没有竞争力?功能测试这活初中生都可以干?功能测试 ...

  9. 基于php编写的新闻类爬虫,插入WordPress数据库

    这个爬虫写的比较久远,很久没有更新博客了. 1.首先思路是:通过php的curl_setopt()函数可以方便快捷的抓取网页. 2.什么样的新闻吸引人呢,当然的热点新闻了.这里选百度的搜索风云榜,获取 ...

  10. 线程池与Python中的GIL

    线程池是一个操作系统的概念,它是对多线程的一种优化. 多线程的时候,创建和销毁线程伴随着操作系统的开销,如果频繁创建/销毁线程,则会使效率大大降低. 而线程池,是先创建出一批线程放入池子里,需要创建线 ...