看到坤哥的博客,自己竟然也出现这个问题了,那就在这进行一下总结学习!
尚学堂的OA 中实现时用Filter来为ThreadLocal对象的PagerSize和Offset设值.这样就可以动态加入值而不用修改manager层的代码逻辑,这是AOP(面向切面编程)的思想.
经本人测试直接把代码移植过来后在Struts1.x中使用没有问题,而在Strus2中使用时出现异常:ognl.OgnlException: target is null for setProperty(null, "offset", [Ljava.lang.String;@c3dfb9),查阅相关资料,有两种方法,第一种方法:在Struts2中加入修改配置webwork.devMode = false;第二种方法是:写一段这样的代码: HttpServletRequest request=ServletActionContext.getRequest();
if(request.getParameter("pager.offset")==null){
offset=0;
}else{
offset=Integer.parseInt(request.getParameter("pager.offset"));
System.out.println("offset:"+offset);
}
经过分析第一种方法显然是在规避问题,根本没有解决.第二种方法加入后代码具有侵入性,这样就失去了AOP编程无侵入性的意义.我现在提供第三种方法:在原来的Action中加入这样一句:public Pager pager=new Pager();建一个Pager类,其中有两个变量pagerSize和offset并实现getter和setter方法;原因很简单提示一下,Sturts2支持Pojo.
本人在测试时发现使用Filter时是不能精确定位到列表显示的方法和页面上的,这样不是造成很多的不需要的重复吗.能不能精确定位到显示列表的方法上呢?
一开始想到了JDK的动态代理来实现,但这样需要在客户端调用时使用代理类.后来想到了使用Spring的AOP编程.这样确实能够实现刚才的想法.后来想到了使用Struts2的拦截器,拦截器是层层嵌套的能和Struts2很好的配合.OK,就使用拦截器了.去掉原来的Filter,改用PagerInterceptor代码如下:
package com.ssh.utile;
import javax.servlet.http.HttpServletRequest;
import org.apache.struts2.ServletActionContext;
import com.opensymphony.xwork2.ActionInvocation;
import com.opensymphony.xwork2.interceptor.MethodFilterInterceptor;
public class PagerInterceptor extends MethodFilterInterceptor {
public static final String PAGE_SIZE_NAME = "ps";
@Override
protected String doIntercept(ActionInvocation invocation) throws Exception {
String result="";
try {
//在调用真正的方法之前在线程内设置pagesize和offset.
HttpServletRequest httpRequest = ServletActionContext.getRequest();
SystemContext.setOffset(getOffset(httpRequest));
SystemContext.setPagesize(getPagesize(httpRequest));
//调用真正的方法.
result=invocation.invoke();
} catch (Exception e) {
e.printStackTrace();
throw new RuntimeException("分页时传入参数出错!");
}finally {
//不管是否调用真正的方法,都要移除线程内的ThreadLocal对象.
SystemContext.removeOffset();
SystemContext.removePagesize();
}
return result;
}
private int getOffset(HttpServletRequest request) {
int offset = 0;
try {
offset = Integer.parseInt(request.getParameter("pager.offset"));
} catch (Exception ignore) {
}
return offset;
}
private int getPagesize(HttpServletRequest httpRequest) {
// 首先判断request中是否有pagesize参数,如果有这个参数,证明客户端正在请求改变每页显示的行数
String psvalue = httpRequest.getParameter(PAGE_SIZE_NAME);
if (psvalue != null && !psvalue.trim().equals("")) {
Integer ps = 0;
try {
ps = Integer.parseInt(psvalue);
} catch (Exception e) {
}
if (ps != 0) {
httpRequest.getSession().setAttribute(PAGE_SIZE_NAME, ps);
}
}
// 判断当前session中是否有pagesize的值
Integer pagesize = (Integer) httpRequest.getSession().getAttribute(PAGE_SIZE_NAME);
if (pagesize == null) {
Integer pageSize= Integer.parseInt(ServletActionContext.getServletContext().getInitParameter("pageSize"));//这里没有用try拦截异常,以为在它的上层拦截了异常.此处进行了每页记录数可配置的扩展功能.
httpRequest.getSession().setAttribute(PAGE_SIZE_NAME, pageSize);
return pageSize;
}
return pagesize;
}
}
在Struts2的配置文件中配置如下:
<interceptors>
<interceptor name="pager" class="com.ssh.utile.PagerInterceptor" />
<interceptor-stack name="myInterceptor">
<interceptor-ref name="pager">
<!--<param name="includeMethods">queryUser</param>-->
</interceptor-ref>
<interceptor-ref name="defaultStack"></interceptor-ref>
</interceptor-stack>
</interceptors>
<action name="user" class="userAction" >
<result name="showQuery">/showQuery.jsp</result>
<result name="error">/error.jsp</result>
<result name="showUpdate">/update.jsp</result>
<result name="success">/pub_add_success.jsp</result>
<interceptor-ref name="myInterceptor"></interceptor-ref>
</action>
在web.xml中加入下面的配置:
<context-param>
<param-name>page_size</param-name>
<param-value>5</param-value>
</context-param>
再加一句,使用Spring的AOP编程也是完全可以实现的.