Struts2拦截器(Interceptor)原理详解

本文详细介绍了Struts2框架中的拦截器概念及其工作原理,包括如何配置和使用内置及自定义拦截器,并通过示例展示了如何实现用户权限控制。

1.    理解拦截器
1.1.    什么是拦截器
拦截器,在AOP(Aspect-Oriented Programming)中用于在某个方法或字段被访问之前,进行拦截然后在之前或之后加入某些操作。拦截是AOP的一种实现策略。
在Webwork的中文文档的解释为——拦截器是动态拦截Action调用的对象。它提供了一种机制可以使开发者可以定义在一个action执行的前后执行的代码,也可以在一个action执行前阻止其执行。同时也是提供了一种可以提取action中可重用的部分的方式。
谈到拦截器,还有一个词大家应该知道——拦截器链(Interceptor Chain,在Struts 2中称为拦截器栈Interceptor Stack)。拦截器链就是将拦截器按一定的顺序联结成一条链。在访问被拦截的方法或字段时,拦截器链中的拦截器就会按其之前定义的顺序被调用。
1.2.    拦截器的实现原理
大部分时候,拦截器方法都是通过代理的方式来调用的。Struts 2的拦截器实现相对简单。当请求到达Struts 2的ServletDispatcher时,Struts 2会查找配置文件,并根据其配置实例化相对的拦截器对象,然后串成一个列表(list),最后一个一个地调用列表中的拦截器。如下图:

2.    拦截器的配置
Struts 2已经为您提供丰富多样的,功能齐全的拦截器实现。大家可以至struts2的jar包内的struts-default.xml查看关于默认的拦截器与拦截器链的配置。
 
       在struts.xml文件中定义拦截器,拦截器栈:
< package   name ="my"   extends ="struts-default"   namespace ="/manage" >  
                 < interceptors >  
                <!--   定义拦截器  -->  
                 < interceptor   name ="拦截器名"   class ="拦截器实现类" />  
                <!--   定义拦截器栈  -->  
                 < interceptor-stack   name ="拦截器栈名" >  
                           < interceptor-ref   name ="拦截器一" />  
                           < interceptor-ref   name ="拦截器二" />  
                 </ interceptor-stack >  
                 </ interceptors >  
                ......  
</ package >  
 
3.    使用拦截器
一旦定义了拦截器和拦截器栈后,就可以使用这个拦截器或拦截器栈来拦截Action了。拦截器的拦截行为将会在Action的exceute方法执行之前被执行。
< action   name ="userOpt"   class ="org.qiujy.web.struts2.action.UserAction" >  
                         < result   name ="success" > /success.jsp </ result >  
                         < result   name ="error" > /error.jsp </ result >  
<!--   使用拦截器,一般配置在result之后,  -->  
<!--   引用系统默认的拦截器  -->  
< interceptor-ref   name ="defaultStack" />  
                         < interceptor-ref   name ="拦截器名或拦截器栈名" />  
                 </ action >  
 
       此处需要注意的是,如果为Action指定了一个拦截器,则系统默认的拦截器栈将会失去作用 。为了继续使用默认拦截器,所以上面配置文件中手动引入了默认拦截器。
4.    自定义拦截器
作为“框架(framework)”,可扩展性是不可或缺的。虽然,Struts 2为我们提供如此丰富的拦截器实现,但是这并不意味我们失去创建自定义拦截器的能力,恰恰相反,在Struts 2自定义拦截器是相当容易的一件事。
  
4.1.    实现拦截器类
所有的Struts 2的拦截器都直接或间接实现接口com.opensymphony.xwork2.interceptor.Interceptor。该接口提供了三个方法:
1)      void init(); 在该拦截器被初始化之后,在该拦截器执行拦截之前,系统回调该方法。对于每个拦截器而言,此方法只执行一次。
2)      void destroy();该方法跟init()方法对应。在拦截器实例被销毁之前,系统将回调该方法。
3)      String intercept(ActionInvocation invocation) throws Exception; 该方法是用户需要实现的拦截动作。该方法会返回一个字符串作为逻辑视图。
除此之外,继承类com.opensymphony.xwork2.interceptor.AbstractInterceptor是更简单的一种实现拦截器类的方式,因为此类提供了init()和destroy()方法的空实现,这样我们只需要实现intercept方法。
4.2.    使用自定义拦截器:两个步骤
l  通过<interceptor …>元素来定义拦截器。
l  通过<interceptor-ref …>元素来使用拦截器。
5.    自定义拦截器示例
5.1.    问题描述
使用自定义拦截器来完成用户权限的控制:当浏览者需要请求执行某个操作时,应用需要先检查浏览者是否登录,以及是否有足够的权限来执行该操作。
5.2.    实现权限控制拦截器类
AuthorizationInterceptor.java
package   org.qiujy.common;  
    
import   java.util.Map;  
    
import   com.opensymphony.xwork2.Action;  
import   com.opensymphony.xwork2.ActionInvocation;  
import   com.opensymphony.xwork2.interceptor.AbstractInterceptor;  
    
/** 
* 权限检查拦截器 
*    
* @author qiujy 
* @version 1.0 
*/
 
public   class   AuthorizationInterceptor   extends   AbstractInterceptor {  
    
         /* 
         * 拦截Action处理的拦截方法 
         *    
         */
 
         public   String intercept(ActionInvocation invocation)   throws   Exception {  
                  
                Map session = invocation.getInvocationContext().getSession();  
                String userName = (String) session.get( "userName" );  
                  
                 if   ( null   != userName && userName.equals( "test" )) {  
                        System.out.println( "拦截器:合法用户登录---" );  
                         return   invocation.invoke();  
                }   else   {  
                        System.out.println( "拦截器:用户未登录---" );  
                         return   Action.LOGIN;  
                }  
        }  
}  
 
 
5.3.    配置权限控制拦截器
struts.xml:
<!DOCTYPE struts PUBLIC  
                "-//Apache Software Foundation//DTD Struts Configuration 2.0//EN"  
                "http://struts.apache.org/dtds/struts-2.0.dtd">  
< struts >  
         < package   name ="my"   extends ="struts-default" >  
                  
                 < interceptors >  
                <!--   定义权限控制拦截器  -->  
                 < interceptor   name ="authority"   class ="org.qiujy.common.AuthorizationInterceptor" />  
                 </ interceptors >  
                  
                <!--   定义全局处理结果  -->  
                 < global-results >  
                <!--   逻辑名为login的结果,映射到/login.jsp页面  -->  
                 < result   name ="login" > /login.jsp </ result >  
                 </ global-results >  
                  
                 < action   name ="listall"   class ="org.qiujy.web.struts2.action.UserAction"   method ="listAllUser" >  
                         < result   name ="success" > /listall.jsp </ result >  
                        <!--   使用拦截器  -->  
                         < interceptor-ref   name ="defaultStack" />  
                         < interceptor-ref   name ="authority" />  
                 </ action >  
                  
                 < action   name ="userOpt"   class ="org.qiujy.web.struts2.action.UserAction" >  
                         < result   name ="success" > /success.jsp </ result >  
                 </ action >  
         </ package >  
</ struts >  
 
    5.4.    运行调试
在浏览器地址栏直接输入 http://localhost:8080/AuthorityInterceptorDemo/listall.action   来访问,此动作配置了权限拦截器,所有被转到登录页面。

登录后:
 

如果为了简化struts.xml文件的配置,避免在每个Action重复配置该拦截器,可以将拦截器配置成了一个默认拦截器栈。如下:
<!DOCTYPE struts PUBLIC  
                "-//Apache Software Foundation//DTD Struts Configuration 2.0//EN"  
                "http://struts.apache.org/dtds/struts-2.0.dtd">  
< struts >  
         < package   name ="my"   extends ="struts-default" >  
    
                 < interceptors >  
                        <!--   定义权限控制拦截器  -->  
                         < interceptor   name ="authority"  
                                 class ="org.qiujy.common.AuthorizationInterceptor"   />  
                        <!--   定义一个包含权限控制的拦截器栈  -->  
                         < interceptor-stack   name ="mydefault" >  
                                 < interceptor-ref   name ="defaultStack"   />  
                                 < interceptor-ref   name ="authority"   />  
                         </ interceptor-stack >  
                 </ interceptors >  
                  
                <!--   定义默认拦截器  -->  
                 < default-interceptor-ref   name ="mydefault"   />  
    
                <!--   定义全局处理结果  -->  
                 < global-results >  
                        <!--   逻辑名为login的结果,映射到/login.jsp页面  -->  
                         < result   name ="login" > /login.jsp </ result >  
                 </ global-results >  
    
                 < action   name ="listall"  
                         class ="org.qiujy.web.struts2.action.UserAction"  
                         method ="listAllUser" >  
                         < result   name ="success" > /listall.jsp </ result >  
                 </ action >  
         </ package >  
          
         < package   name ="font"   extends ="struts-default" >  
                 < action   name ="userOpt"   class ="org.qiujy.web.struts2.action.UserAction" >  
                         < result   name ="success" > /success.jsp </ result >  
                 </ action >  
         </ package >  
</ struts >  
 
一旦在某个包下定义了默认拦截器栈,在该包下的所有action都会使用此拦截器栈。对于那些不想使用些拦截器栈的action,则应该将它放置在其它的包下。

本文来自优快云博客,转载请标明出处: http://blog.youkuaiyun.com/qjyong/archive/2007/10/14/1824607.aspx

下载前可以先看下教程 https://pan.quark.cn/s/a4b39357ea24 在网页构建过程中,表单(Form)扮演着用户与网站之间沟通的关键角色,其主要功能在于汇集用户的各类输入信息。 JavaScript作为网页开发的核心技术,提供了多样化的API和函数来操作表单组件,诸如input和select等元素。 本专题将详细研究如何借助原生JavaScript对form表单进行视觉优化,并对input输入框与select下拉框进行功能增强。 一、表单基础1. 表单组件:在HTML语言中,<form>标签用于构建一个表单,该标签内部可以容纳多种表单组件,包括<input>(输入框)、<select>(下拉框)、<textarea>(多行文本输入区域)等。 2. 表单参数:诸如action(表单提交的地址)、method(表单提交的协议,为GET或POST)等属性,它们决定了表单的行为特性。 3. 表单行为:诸如onsubmit(表单提交时触发的动作)、onchange(表单元素值变更时触发的动作)等事件,能够通过JavaScript进行响应式处理。 二、input元素视觉优化1. CSS定制:通过设定input元素的CSS属性,例如border(边框)、background-color(背景色)、padding(内边距)、font-size(字体大小)等,能够调整其视觉表现。 2. placeholder特性:提供预填的提示文字,以帮助用户明确输入框的预期用途。 3. 图标集成:借助:before和:after伪元素或者额外的HTML组件结合CSS定位技术,可以在输入框中嵌入图标,从而增强视觉吸引力。 三、select下拉框视觉优化1. 复选功能:通过设置multiple属性...
【EI复现】基于深度强化学习的微能源网能量管理与优化策略研究(Python代码实现)内容概要:本文围绕“基于深度强化学习的微能源网能量管理与优化策略”展开研究,重点探讨了如何利用深度强化学习技术对微能源系统进行高效的能量管理与优化调度。文中结合Python代码实现,复现了EI级别研究成果,涵盖了微电网中分布式能源、储能系统及负荷的协调优化问题,通过构建合理的奖励函数与状态空间模型,实现对复杂能源系统的智能决策支持。研究体现了深度强化学习在应对不确定性可再生能源出力、负荷波动等挑战中的优势,提升了系统运行的经济性与稳定性。; 适合人群:具备一定Python编程基础和机器学习背景,从事能源系统优化、智能电网、强化学习应用等相关领域的研究生、科研人员及工程技术人员。; 使用场景及目标:①应用于微能源网的能量调度与优化控制,提升系统能效与经济效益;②为深度强化学习在能源管理领域的落地提供可复现的技术路径与代码参考;③服务于学术研究与论文复现,特别是EI/SCI级别高水平论文的仿真实验部分。; 阅读建议:建议读者结合提供的Python代码进行实践操作,深入理解深度强化学习算法在能源系统建模中的具体应用,重点关注状态设计、动作空间定义与奖励函数构造等关键环节,并可进一步扩展至多智能体强化学习或与其他优化算法的融合研究。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值