拦截器

1. 拦截器

1.1 概述

ü  基本概念
Intercetor, 即为拦截器。
1) 在Struts2中,把每一个功能都用一个个的拦截器实现;用户想用struts的哪个功能的时候,可以自由组装使用。
2)Struts2中,为了方法用户对拦截器的引用,提供了拦截器栈的定义,里面可以包含多个拦截器。   文件夹(文件, 文件2)  拦截器栈(拦截器,拦截器2)
3)Struts2中,如果用户没有指定执行哪些拦截器,struts2有一个默认执行的栈,defaultStack;
           一旦如果用户有指定执行哪些拦截器,默认的拦截器栈就不会被执行
 
拦截器的设计,就是基于组件设计的应用!
 
ü  拦截器配置举例
 
struts-default.xml文件中,定义了struts提供的所有拦截器!
 
 
//1. 定义拦截器以及拦截器栈
<interceptors>
    1.1 拦截器定义
    <interceptor name="" class="" />
   
    1.2 拦截器栈的定义
    <interceptor-stack name="defaultStack">
         引用了上面拦截器(1.1)
    </interceptor-stack>
</interceptors>
 
2. 默认执行的拦截器(栈)
<default-interceptor-ref name="defaultStack"/>
 
ü  API
|-- Interceptor  拦截器接口
   |-- AbstractInterceptor  拦截器默认实现的抽象类;一般用户只需要继承此类即可继续拦截器开发
 
|-- ActionInvocation拦截器的执行状态,调用下一个拦截器或Action
 
 
拦截器与过滤器区别:
         ….
        
 
1.2 自定义一个拦截器案例

步骤:
         1.写拦截器类 (看生命周期)
         2.配置
 
 
 
/**
 * 自定义拦截器
 * @author Jie.Yuan
 *
 */
public class HelloInterceptor implements Interceptor{
  
   // 启动时候执行
   public HelloInterceptor(){
     System.out.println("创建了拦截器对象");
   }
 
   // 启动时候执行
   @Override
   public void init() {
     System.out.println("执行了拦截器的初始化方法");
   }
 
   // 拦截器业务处理方法(在访问action时候执行?在execute之前执行?)
   @Override
   public String intercept(ActionInvocation invocation) throws Exception {
     System.out.println("2. 拦截器,业务处理-开始");
    
     // 调用下一个拦截器或执行Action  (相当于chain.doFilter(..)
     // 获取的是: execute方法的返回值
     String resultFlag = invocation.invoke();
    
     System.out.println("4. 拦截器,业务处理-结束");
    
     return resultFlag;
   }
  
 
   @Override
   public void destroy() {
     System.out.println("销毁....");
   }
 
 
}
<?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>
   <package name="hello" extends="struts-default">
  
     <!-- 【拦截器配置】 -->
     <interceptors>
    
        <!-- 配置用户自定义的拦截器 -->
        <interceptor name="helloInterceptor" class="cn.itcast.a_interceptor.HelloInterceptor"></interceptor>
       
        <!-- 自定义一个栈:要引用默认栈、自定义的拦截器 -->
        <interceptor-stack name="helloStack">
           <!-- 引用默认栈 (一定要放到第一行)-->
           <interceptor-ref name="defaultStack"></interceptor-ref>
           <!-- 引用自定义拦截器 -->
           <interceptor-ref name="helloInterceptor"></interceptor-ref>
        </interceptor-stack>
       
     </interceptors>
    
     <!-- 【执行拦截器】 -->
     <default-interceptor-ref name="helloStack"></default-interceptor-ref>
    
  
     <!-- Action配置 -->
     <action name="hello" class="cn.itcast.a_interceptor.HelloAction">
        <result name="success"></result>
     </action>
  
  
   </package>
</struts>
 
 
 
1.2 拦截器执行流程

 
UML (时序图)
启动:
         创建所有拦截器、执行init()
访问:
         先创建Action,
         再执行拦截器,
         最后:拦截器放行,执行execute();
 
 
1.3 拦截器案例

需求:
         登陆后,显示列表!
案例准备:
         Strutsjar文件
         DbUtils组件
         数据库连接池/ 驱动包
        
 
--> login.jsp
<body>
     <form method="post" action="${pageContext.request.contextPath }/user_login.action">
        用户名:<input type="text" name="admin.userName"><br/>
        密码:<input type="text" name="admin.pwd"><br/>
        <input type="submit" value="登陆"><br/>
     </form>
  </body>
à UserAction.java
 
public class UserAction extends ActionSupport {
 
    // ---------1. 封装请求数据-----------
    private Admin admin;
    public Admin getAdmin() {
        return admin;
    }
    public void setAdmin(Admin admin) {
        this.admin = admin;
    }
    // ---------2. 调用的Service-----------
    private AdminService adminService = new AdminService();
   
    // 登陆
    public String login() {
        try {
            Admin userInfo = adminService.login(admin);
            // 判断
            if (userInfo == null){
                // 登陆失败
                return "input";
            }
            // 登陆成功:数据保存在session中
            ActionContext.getContext().getSession().put("userInfo", userInfo);
           
            // 登陆成功
            return "loginSuccess";
        } catch (Exception e) {
            return ERROR;
        }
    }
   
    // 列表
    public String list() {
        try {
            // 查询全部
            List<Admin> list = adminService.getAll();
            // 保存到request
            ActionContext.getContext().getContextMap().put("listAdmin", list);
            return "list";
        } catch (Exception e) {
            return ERROR;
        }
    }
   
    public String add() {
        return null;
    }
   
   
}
à list.jsp
<body>
    <h1>欢迎你,${userInfo.userName }</h1>
    <table align="center" border="1">
        <tr>
            <td>序号</td>
            <td>编号</td>
            <td>用户名</td>
            <td>密码</td>
        </tr>
        <%--@taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c" --%>
        <!-- 用struts标签迭代数据 -->
        <%@taglib uri="/struts-tags" prefix="s" %>
        <s:iterator var="admin" value="#request.listAdmin" status="st">
            <tr>
                <td>
                    <s:property value="#st.count"/>
                </td>
                <td>
                    <s:property value="#admin.id"/>
                </td>
                <td>
                    <s:property value="#admin.userName"/>
                </td>
                <td>
                    <s:property value="#admin.pwd"/>
                </td>
            </tr>
        </s:iterator>
       
    </table>
  </body>
à 自定义拦截器
 
public class UserCheckInterceptor extends AbstractInterceptor{
 
    /**
     * 拦截器业务处理方法
     */
    public String intercept(ActionInvocation invocation) throws Exception {
        // 拿到当前执行的方法名:判断,只有当前方法名不是login,就进行验证
       
        // 获取ActionContext对象
        ActionContext ac = invocation.getInvocationContext();
       
        // 获取action的代理对象
         ActionProxy proxy = invocation.getProxy();
         // 获取当前执行的方法名
         String methodName = proxy.getMethod();
         // 判断
         if (!"login".equals(methodName)) {
             // 先获取当前登陆的用户
             Object obj = ac.getSession().get("userInfo");
             if (obj == null) {
                 // 没有登陆
                 return "input";
             } else {
                 // 当前用户有登陆
                 return invocation.invoke();
             }
         } else {
             // 说明当前用户正在登陆
             return invocation.invoke();
         }
    }
 
}
à 配置拦截器
<?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>
    <package name="user" extends="struts-default">
   
        <!-- 【拦截器配置】 -->
        <interceptors>
            <interceptor name="loginCheck" class="cn.itcast.interceptor.UserCheckInterceptor"></interceptor>
            <interceptor-stack name="myStack">
                <interceptor-ref name="defaultStack"></interceptor-ref>
                <interceptor-ref name="loginCheck"></interceptor-ref>
            </interceptor-stack>
        </interceptors>
        <!-- 【执行拦截器:第一种写法:当前包下所有的acntion都执行myStack栈】
        <default-interceptor-ref name="myStack"></default-interceptor-ref>
         -->
   
        <!-- 全局配置 -->
        <global-results>
            <result name="error">/error.jsp</result>
        </global-results>
       
        <action name="user_*" class="cn.itcast.action.UserAction" method="{1}">
           
            <!--第二种写法:只是在这一个Action中执行myStack栈
            <interceptor-ref name="defaultStackt"></interceptor-ref>
            <interceptor-ref name="loginCheck"></interceptor-ref>
            -->
           
            <!-- 第三种写法:执行用户栈(与第二种写法一样,只在当前aciton中执行自定义栈) -->
            <interceptor-ref name="myStack"></interceptor-ref>
           
           
           
            <!-- 1. 登陆失败 -->
            <result name="input">/login.jsp</result>
           
            <!-- 2. 登陆成功 -->
            <result name="loginSuccess" type="redirectAction">user_list</result>
           
            <!-- 3. 列表展示 -->
            <result name="list">/WEB-INF/list.jsp</result>
           
        </action>
       
    </package>
</struts>
 
 
 
 
2. Struts2中的国际化

回顾:Servlet 中国际化:
         1.写资源文件
                  基础名.properties  【默认的语言环境的配置】 
                  基础名_语言简称_国家简称.properties
         2.读取资源文件,再使用
                  程序:ResourceBundle
                  Jsp:   jstl提供的格式化与国际化标签库。
 
 
 
Struts2中国际化:
         1.写资源文件  (同servlet)
         2.读资源文件
                  程序:ResourceBundle   (同servlet)
                  JSP: 
                          1)jstl表亲啊  (同servlet)
                          2)struts标签获取资源文件内容
 
区别:
         Struts2加载资源文件更加简单!通过常量加载即可!再在jsp页面直接使用!
 
 
à1.  写资源文件
Msg.properties   默认的语言环境; 找不到配置就找它
Msg_en_US.properties  美国
-à2.  加载
<constant name="struts.custom.i18n.resources" value="cn.itcast.config.msg"></constant>
à3. 使用: 标签name值直接写配置文件中的key
<s:text name="title"></s:text>
 
 
另外一点,
         (推荐)加载资源文件通过常量加载
         还可以在页面加载, 这样用:
                  <s:i18nname="cn.itcast.config.msg">
                          <s:text>  标签必须放到标签体中。
</s:i18n>
 
 
3. Ognl表达式语言

概述

l  OGNL表达式
      OGNL是Object Graphic Navigation Language(对象图导航语言)的缩写,它是一个开源项目。 Struts2框架使用OGNL作为默认的表达式语言。
l  OGNL优势
     1、支持对象方法调用,如xxx.doSomeSpecial();
      2、支持类静态的方法调用和值访问,表达式的格式:
            @[类全名(包括包路径)]@[方法名 |  值名],例如:
            @java.lang.String@format('foo %s', 'bar')
            或@tutorial.MyConstant@APP_NAME;
      3、支持赋值操作和表达式串联,如price=100, discount=0.8,
            calculatePrice(),这个表达式会返回80;
      4、访问OGNL上下文(OGNL context)和ActionContext;
      5、操作集合对象。
l  总结
        OGNL 有一个上下文(Context)概念,说白了上下文就是一个MAP结构,它实现了  java.utils.Map 的接口。 OgnlContext对象
 
 
分析:
ü  Struts框架默认就支持Ognl表达式语言。
(struts必须引用的包:ognl.jar)
ü  作用
页面取值用。
 
El表达式语言,用于页面取值,jsp页面取值的标准。(默认直接可以使用)
         (应用范围更广。)
Ognl表达式语言, struts标签默认支持的表达式语言。
                                     必须配置struts标签用,不能离开struts标签直接用。
 
OgnlContext对象(了解)

OgnlContext对象是ognl表达式语言的核心。
源码类:
public class OgnlContext extends Objectimplements Map{..}
 
硬编码方式,了解OgnlContext对象:
 
 
// OgnlContext用法
public class OgnlDemo1 {
 
    /**
     * 1. Ognl表达式语言语言取值,取非根元素的值,必须用#号
     * @throws Exception
     */
    @Test
    public void testOgnl() throws Exception {
        // 创建一个Ognl上下文对象
        OgnlContext context = new OgnlContext();
        // 放入数据
        User user = new User();
        user.setId(100);
        user.setName("Jack");
        // 【往非根元素放入数据,取值的时候表达式要用"#"】
        context.put("user", user);
       
        // 获取数据(map)
        // 先构建一个Ognl表达式, 再解析表达式
        Object ognl = Ognl.parseExpression("#user.name");
        Object value = Ognl.getValue(ognl, context, context.getRoot());
       
        System.out.println(value);
    }
   
    /**
     * 2. Ognl表达式语言语言取值,取根元素的值,不用带#号
     * @throws Exception
     */
    @Test
    public void testOgn2() throws Exception {
        // 创建一个Ognl上下文对象
        OgnlContext context = new OgnlContext();
        // 放入数据
        User user = new User();
        user.setId(100);
        user.setName("Jack");
        // 【往根元素放入数据】
        context.setRoot(user);
       
        // 获取数据(map)
        // 先构建一个Ognl表达式, 再解析表达式
        Object ognl = Ognl.parseExpression("address.province");
        Object value = Ognl.getValue(ognl, context, context.getRoot());
       
        System.out.println(value);
    }
   
    /**
     * 3.Ognl对静态方法调用的支持
     * @throws Exception
     */
    @Test
    public void testOgn3() throws Exception {
        // 创建一个Ognl上下文对象
        OgnlContext context = new OgnlContext();
       
        // Ognl表单式语言,调用类的静态方法
        //Object ognl = Ognl.parseExpression("@Math@floor(10.9)");
        // 由于Math类在开发中比较常用,所以也可以这样写
        Object ognl = Ognl.parseExpression("@@floor(10.9)");
        Object value = Ognl.getValue(ognl, context, context.getRoot());
        System.out.println(value);
    }
}
 
 
 
ValueStack对象

ValueStack, 即值栈对象。
值栈对象:
         是整个struts数据存储的核心,或者叫中转站。
         用户每次访问struts的action,都会创建一个Action对象、值栈对象、ActionContext对象;  然后把Action对象放入值栈中; 最后再把值栈对象放入request中,传入jsp页面。
(key: struts.valueStack);     开发者只需要通过ActionContext对象就可以访问struts的其他的关键对象。 (ActionContext是给开发者用的,便于学习与使用。)
 
 
问题:
         OgnlContext与ValueStack对象的关系?
 
Struts标签

Struts标签取值,就使用了Ognl表达式语言。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值