JavaWeb中Struts2拦截器深入分析(一)
一、struts2中的拦截器(框架功能核心)
1、过滤器vs拦截器
过滤器vs拦截器功能是一回事。过滤器是servlet规范中的技术,可以对请求和响应进行过滤。
拦截器是struts2框架中的技术,实现aop(面向切面)的编程思想,是可插拔的, 可以对访问某个 action 方法之前或之后实施拦截。
拦截器栈(interceptor stack): 将拦截器按一定的顺序联结成一条链. 在访问被拦截的方法时, struts2拦截器链中的拦截器就会按其之前定义的顺序被依次调用
struts2执行原理 - 底层分析
2、自定义拦截器
struts2定义了一个拦截器接口interceptor接口。
interceptor接口里面有三个抽象方法
•init: 该方法将在拦截器被创建后立即被调用, 它在拦截器的生命周期内只被调用一次. 可以在该方法中对相关资源进行必要的初始化
•interecept: 每拦截一个动作请求, 该方法就会被调用一次.
•destroy: 该方法将在拦截器被销毁之前被调用, 它在拦截器的生命周期内也只被调用一次.
struts 会依次调用程序员为某个 action 而注册的每一个拦截器的 interecept 方法.每次调用 interecept 方法时, struts 会传递一个 actioninvocation 接口的实例.
actioninvocation: 代表一个给定动作的执行状态, 拦截器可以从该类的对象里获得与该动作相关联的 action 对象和 result 对象. 在完成拦截器自己的任务之后, 拦截器将调用 actioninvocation 对象的 invoke 方法前进到 action 处理流程的下一个环节.
还可以调用 actioninvocation 对象的 addpreresultlistener 方法给 actioninvocation 对象 “挂” 上一个或多个 preresultlistener 监听器. 该监听器对象可以在动作执行完毕之后, 开始执行动作结果之前做些事情
自定义拦截器步骤:
a、编写一个类,实现com.opensymphony.xwork2.interceptor.interceptor接口,或者继承
com.opensymphony.xwork2.interceptor.abstractinterceptor类。(适配器模式),一般都选择继承abstractinterceptor(拦截器会驻留内存)。因为abstractinterceptor 类实现了 interceptor 接口. 并为 init, destroy 提供了一个空白的实现
编写两个拦截器interceptordemo1 ,和interceptordemo2
package com.itheima.interceptor; import com.opensymphony.xwork2.actioninvocation; import com.opensymphony.xwork2.interceptor.abstractinterceptor; public class interceptordemo1 extends abstractinterceptor { //动作的每次访问都会调用该方法 public string intercept(actioninvocation invocation) throws exception { system.out.println("拦截前demo1"); string rtvalue = invocation.invoke();//放行,这里为什么返回string? 因为最终的结果返回的action的result,而action的结果是string类型 system.out.println("拦截后demo1"); return rtvalue; } } package com.itheima.interceptor; import com.opensymphony.xwork2.actioninvocation; import com.opensymphony.xwork2.interceptor.abstractinterceptor; import com.opensymphony.xwork2.interceptor.preresultlistener; public class interceptordemo2 extends abstractinterceptor { //动作的每次访问都会调用该方法 public string intercept(actioninvocation invocation) throws exception { // invocation.addpreresultlistener(new preresultlistener() { // // public void beforeresult(actioninvocation invocation, string resultcode) { // system.out.println("结果显示前"); // } // }); system.out.println("拦截前demo2"); string rtvalue = invocation.invoke();//放行 system.out.println("拦截后demo2"); return rtvalue; } }
b、需要在struts.xml中进行定义,定义拦截器,先定义在使用。
<package name="p1" extends="struts-default"> <!-- 定义拦截器:只对当前包有效 --> <interceptors> <interceptor name="interceprotdemo1" class="com.itheima.interceptor.interceptordemo1"></interceptor> <interceptor name="interceprotdemo2" class="com.itheima.interceptor.interceptordemo2"></interceptor> </interceptors> </package>
c、在动作配置中就可以使用了
<action name="action1" class="com.itheima.action.demo1action" method="execute"> <!-- 使用定义的拦截器。如过没有指定任何的拦截器,默认使用default-stack栈中的所有拦截器; 一旦指定了任何一个拦截器,默认的就无效了 --> <interceptor-ref name="interceprotdemo1"></interceptor-ref> <interceptor-ref name="interceprotdemo2"></interceptor-ref> <result>/success.jsp</result> </action>
实现动作类demo1action
package com.itheima.action; import com.opensymphony.xwork2.actionsupport; public class demo1action extends actionsupport { @override public string execute() throws exception { system.out.println("execute执行了"); return success; } }
运行结果
因为struts2中如文件上传,数据验证,封装请求参数到action等功能都是由系统默认的defaultstack中的拦截器实现的,所以我们定义的拦截器需要引用系统默认的defaultstack,这样应用才可以使用struts2框架提供的众多功能。
如过没有指定任何的拦截器,默认使用default-stack栈中的所有拦截器;一旦指定了任何一个拦截器,默认的就无效了除了要使用自定义的拦截器之外,还要使用defaultstack,可以这么办
方法一:(自己使用),只需在action中配置自定义的和defaultstack默认的就可以了。
方法二:(大家都用的时候),如果希望包下的所有action都使用自定义的拦截器, 要使用拦截器栈 interceptor-stack,定义一个interceptor-stack,然后在action中可以通过<default-interceptor-ref name=“mydefaultstack”/>把拦截器定义为默认拦截器,mydefaultstack名字可以自己取。
<interceptors> <interceptor name="interceprotdemo1" class="com.itheima.interceptor.interceptordemo1"></interceptor> <interceptor name="interceprotdemo2" class="com.itheima.interceptor.interceptordemo2"></interceptor> <interceptor-stack name="mydefaultstack"> <interceptor-ref name="defaultstack"></interceptor-ref> <interceptor-ref name="interceprotdemo1"></interceptor-ref> <interceptor-ref name="interceprotdemo2"></interceptor-ref> </interceptor-stack> </interceptors> <action name="action3" class="com.itheima.action.loginaction" method="login"> <interceptor-ref name="mydefaultstack"></interceptor-ref> <result>/success.jsp</result> </action>
3、struts2 自带的拦截器
案例1:检查用户是否登录
1、 编写页面login.jsp
<body> <form action="${pagecontext.request.contextpath}/login.action" method="post"> <input type="text" name="username"/><br/> <input type="text" name="password"/><br/> <input type="submit" value="登录"/> </form> </body>
2、编写登录校验的拦截器logincheckinterceptor 类
package com.itheima.interceptor; import javax.servlet.http.httpsession; import org.apache.struts2.servletactioncontext; import com.opensymphony.xwork2.actioninvocation; import com.opensymphony.xwork2.interceptor.abstractinterceptor; public class logincheckinterceptor extends abstractinterceptor { public string intercept(actioninvocation invocation) throws exception { httpsession session = servletactioncontext.getrequest().getsession();//通过servletactioncontext对象获得session对象 object user = session.getattribute("user"); if(user==null){ //没有登录 return "login";//返回到某个逻辑视图 } return invocation.invoke();//放行 } }
3、编写配置文件struts.xml
<package name="p2" extends="struts-default"> <interceptors> <interceptor name="logincheckinterceptor" class="com.itheima.interceptor.logincheckinterceptor"></interceptor> <interceptor-stack name="mydefaultstack"> <interceptor-ref name="defaultstack"></interceptor-ref> <interceptor-ref name="logincheckinterceptor"></interceptor-ref> </interceptor-stack> </interceptors> <action name="login" class="com.itheima.action.customeraction" method="login"> <result>/login.jsp</result> </action> </package>
4、编写动作类customeraction
package com.itheima.action; import org.apache.struts2.servletactioncontext; import com.opensymphony.xwork2.actionsupport; public class customeraction extends actionsupport { public string login(){ system.out.println("登录"); servletactioncontext.getrequest().getsession().setattribute("user", "ppp"); return success; } }
案例2:监测动作方法的执行效率
编写时间监测过滤器timerinterceptor
package com.itheima.interceptor; import com.opensymphony.xwork2.actioninvocation; import com.opensymphony.xwork2.interceptor.abstractinterceptor; public class timerinterceptor extends abstractinterceptor { public string intercept(actioninvocation invocation) throws exception { long time = system.nanotime(); string rtvalue = invocation.invoke(); system.out.println(rtvalue+"执行耗时:"+(system.nanotime()-time)+"纳秒"); return rtvalue; } }
编写配置文件
<package name="p2" extends="struts-default"> <interceptors> <interceptor name="logincheckinterceptor" class="com.itheima.interceptor.logincheckinterceptor"></interceptor> <interceptor name="timerinterceptor" class="com.itheima.interceptor.timerinterceptor"></interceptor> <interceptor-stack name="mydefaultstack"> <interceptor-ref name="defaultstack"></interceptor-ref> <interceptor-ref name="logincheckinterceptor"></interceptor-ref> <interceptor-ref name="timerinterceptor"></interceptor-ref> </interceptor-stack> </interceptors> <result name="login">/login.jsp</result> </action> </package>
从上面可以看出,在一个action 中可以配置多个过滤器。
4、自定义拦截器:能够指定拦截的方法或不拦截的方法
能够指定拦截的方法或不拦截的方法,编写过滤器时,可以实现类methodfilterinterceptor,里面有两个字段,通过注入参数就可以指定那些不拦截,两个参数只要用一个即可,当拦截较少是,可以用includemethods ,当拦截较多是,可以用排除的方法excludemethods 。
excludemethods = collections.emptyset();//排除那些
includemethods = collections.emptyset();//包括那些
案例:再续登录校验的例子。
1、编写过滤器logincheckinterceptor
package com.itheima.interceptor; import javax.servlet.http.httpsession; import org.apache.struts2.servletactioncontext; import com.opensymphony.xwork2.actioninvocation; import com.opensymphony.xwork2.interceptor.abstractinterceptor; import com.opensymphony.xwork2.interceptor.methodfilterinterceptor; public class logincheckinterceptor extends methodfilterinterceptor { protected string dointercept(actioninvocation invocation) throws exception { httpsession session = servletactioncontext.getrequest().getsession(); object user = session.getattribute("user"); if(user==null){ //没有登录 return "login";//返回到某个逻辑视图 } return invocation.invoke();//放行 } }
2、编写配置文件
3、编写动作类customeraction
package com.itheima.action; import org.apache.struts2.servletactioncontext; import com.opensymphony.xwork2.actionsupport; public class customeraction extends actionsupport { public string add(){ system.out.println("调用add的service方法"); return success; } public string edit(){ system.out.println("调用edit的service方法"); return success; } public string login(){ system.out.println("登录"); servletactioncontext.getrequest().getsession().setattribute("user", "ppp"); return success; } }
4、编写页面
addcustomer.jsp
<body> 添加客户 </body>
editcustomer.jsp
<body> 修改客户 </body>
login.jsp
<body> <form action="${pagecontext.request.contextpath}/login.action" method="post"> <input type="text" name="username"/><br/> <input type="text" name="password"/><br/> <input type="submit" value="登录"/> </form> </body>
success.jsp
<body> oyeah </body>
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。
推荐阅读
-
JavaWeb中Struts2拦截器深入分析(一)
-
JavaWeb中struts2实现文件上传下载功能实例解析
-
JavaWeb中struts2实现文件上传下载功能实例解析
-
java中的Struts2拦截器详解
-
java中的Struts2拦截器详解
-
springMVC拦截器从Request中获取Json格式并解决request的请求流只能读取一次的问题
-
SpringBoot 之 SpringMVC拦截器从Request中获取参数并解决request的请求流只能读取一次的问题
-
struts2中一个表单中提交多个请求的例子(多个提交按钮)
-
(原创)超详细一步一步在eclipse中配置Struts2环境,无基础也能看懂
-
struts2中的拦截器详解。