Struts2拦截器概述
1 Struts2是框架,封装了很多功能,struts2里面封装的概念都是在拦截器里面
2 Struts2里面封装了很多的概念,有很多拦截器,不是每次这些拦截器都执行,每次执行默认的拦截器
3 Struts2里面默认拦截器位置
拦截器示例代码:
<interceptor-stack name="defaultStack"> <interceptor-ref name="exception"/> <interceptor-ref name="alias"/> <interceptor-ref name="servletConfig"/> <interceptor-ref name="i18n"/> <interceptor-ref name="prepare"/> <interceptor-ref name="chain"/> <interceptor-ref name="debugging"/> <interceptor-ref name="profiling"/> <interceptor-ref name="scopedModelDriven"/> <interceptor-ref name="modelDriven"/> <interceptor-ref name="fileUpload"/> <interceptor-ref name="checkbox"/> <interceptor-ref name="staticParams"/> <interceptor-ref name="actionMappingParams"/> <interceptor-ref name="params"> <param name="excludeParams">dojo\..*,^struts\..*</param> </interceptor-ref> <interceptor-ref name="conversionError"/> <interceptor-ref name="validation"> <param name="excludeMethods">input,back,cancel,browse</param> </interceptor-ref> <interceptor-ref name="workflow"> <param name="excludeMethods">input,back,cancel,browse</param> </interceptor-ref> </interceptor-stack>
4 拦截器在什么时候进行?
(1)在action对象创建之后,action方法之前执行。
拦截器底层原理
1 拦截器底层使用两个原理
第一个aop思想
(1)文字描述
Aop是面向切面编程,有基本概念,扩展功能,不通过修改业务处理源代码方式扩展功能。
(2)画图分析
第二个责任链模式
(1)在java中有很多的设计模式,责任链模式是其中的一种
(2)责任链模式和过滤链很相似的
责任链模式:
要执行多个操作,有添加、修改、删除三个操作。
首先执行添加操作,添加操作执行之后做类似于放行操作,执行修改操作,修改操作执行之后做类似放行操作,执行删除操作。
过滤链:一个请求可以有多个过滤器进行过滤,每个过滤只有做放行才能到下一个过滤器
2 aop思想和责任链模式如何应用到拦截器里面?
(1)文字描述
-拦截器在action对象创建之后,action的方法执行之前执行
-在action方法执行之前执行默认拦截器,执行过程使用aop思想,在action没有直接调用拦截器的方法,使用配置文件进行操作。
-在执行拦截器时候,执行很多的拦截器,这个过程使用责任链模式
-假如要执行三个拦截器,执行拦截器1,执行拦截器1之后作放行操作,执行拦截器2,执行拦截器2之后放行,执行拦截器3之后放行,执行action方法。
Struts2默认会执行很多拦截器,这和配置有关。
拦截器执行过程:
3 查看源代码
(1)执行action
(2)创建action对象,使用动态代理方式
(3)执行action的方法
(4)执行很多的拦截器,遍历执行
类似于放行的操作的方法。
重要的概念
1 过滤器和拦截器的区别
(1)过滤器:过滤器理论上可以过滤任意内容,比如html,jsp,servlet,图片路径
(2)拦截器:拦截器可以拦截action
2 servlet和action区别
(1)servlet默认是第一次访问时创建,创建一次,单实例对象
(2)action每次访问时候创建,创建多次,多实例对象。
自定义拦截器
1 在struts2里面有很多的拦截器,这些拦截器是struts2封装的功能,但是在实际开发中,
struts2里面的拦截器中可以没有我们要使用的功能,这个时候需要自己写拦截器实现功能。
(1)查看源代码看拦截器结构
-继承类
在接口里面有三个方法
做初始化操作
销毁
拦截逻辑操作
(2)开发中,建议使用另外一种方式
-写类,继承MethodFilterInterceptor类实现
-让action里面的某个方法不进行拦截
(3)让拦截器和action有关系
-不是在action调用拦截器的方法,而是通过配置文件方式让建立关系
自定义登录拦截器
1 需求:在项目中,有很多action的超链接,实现只有是登录的状态,才可以点击action的超链接实现功能,如果不是登录功能,点击action超链接返回登录页面。
2 登录的状态:使用session域对象实现
(1) 登录成功之后,把数据放到session里面
(2)判断session是否有值,可以知道是否是登录状态
3 实现登录的基本功能
(1)查询数据库判断用户名和密码
4 添加登录拦截器功能
(1)判断是否登录:判断session里面是否有名称是username的值
(2)拦截器实现过程
第一步 创建类,继承MethodFilterInterceptor类
第二步 重新MethodFilterInterceptor类里面的方法写拦截器逻辑
public class LoginInterceptor extends MethodFilterInterceptor { //这个方法里面写拦截器逻辑 protected String doIntercept(ActionInvocation invocation) throws Exception { //判断session里面是否有名称是username的值 //得到session HttpServletRequest request = ServletActionContext.getRequest(); Object obj = request.getSession().getAttribute("username"); //判断 if(obj != null){ //登录操作 //做类似于放行操作,执行action的方法 return invocation.invoke(); }else{ //不是登录状态 //不到登录,不执行action的方法,返回登录页面 //到result标签里面找到名称是login的值,到配置路径里面 return "login"; } } }
第三步 配置action和拦截器关系(注册拦截器)
action没有调用interceptor的语句,而是通过配置,也叫注册拦截器。
(1)在要拦截的action标签所在的package标签里面声明拦截器
<interceptors> <interceptor name="loginintercept" class="cn.itcast.interceptor.LoginInterceptor" /> </interceptors>
(2)在具体的action标签里面使用声明的拦截器
<interceptor-ref name="loginintercept"></interceptor-ref>
(3)struts2里面执行很多的默认拦截器,但是如果在action里面配置自定义拦截器。
问题:默认的拦截器不会执行了。
解决:把默认拦截器手动使用一次
<!-- 3 把默认拦截器手动使用一次 --> <interceptor-ref name="defaultStack"></interceptor-ref>
-配置拦截器,对action里面所有的方法都进行拦截。