拦截器 & Structs国际化 & Ognl

拦截器

  • 拦截器概述 :

    • 基本概念 : 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. 配置
      3. 代码示例
// 自定义拦截器
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>
  • 拦截器执行流程

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

    • 需求:登陆后,显示列表!
    • 案例准备:
      • Struts jar文件
      • 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>

Structs国际化

  • Struts2中国际化:
    1. 写资源文件 (同servlet)
    2. 读资源文件
    • 程序:ResourceBundle (同servlet)
    • JSP:
      1)jstl标签(同servlet)
      2)struts标签获取资源文件内容
  • 具体步骤 :
    1. 写资源文件 :
    • Msg.properties : 默认的语言环境, 找不到配置就找它
    • Msg_en_US.properties : 美国
    1. 加载 : <constant name="struts.custom.i18n.resources" value="cn.itcast.config.msg"></constant>
    2. 使用 : 标签name值直接写配置文件中的key : <s:text name="title"></s:text>
    • 另外一点,(推荐)加载资源文件通过常量加载还可以在页面加载, 这样用:
<s:i18n name="cn.itcast.config.msg">
    <s:text>  标签必须放到标签体中 </s:text>
</s:i18n>

Ognl表达式语言

  • 概述 : OGNL是Object Graphic Navigation Language(对象图导航语言)的缩写, 它是一个开源项目, Struts2框架使用OGNL作为默认的表达式语言

    • 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、操作集合对象
    • 总结 : OGNL 有一个上下文(Context)概念,说白了上下文就是一个MAP结构, 它实现了 java.utils.Map 的接口。 OgnlContext对象
    • 分析:Struts框架默认就支持Ognl表达式语言(struts必须引用的包:ognl.jar)
    • 作用 : 页面取值用
      1> El表达式语言 : jsp页面取值的标准。(默认直接可以使用) (应用范围更广)
      2> Ognl表达式语言 : struts标签默认支持的表达式语言。必须配置struts标签用, 不能离开struts标签直接用
  • OgnlContext对象 : OgnlContext对象是ognl表达式语言的核心。

    • 源码类:public class OgnlContext extends Object implements 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(即值栈对象) :
    • 值栈对象:是整个struts数据存储的核心,或者叫中转站。用户每次访问struts的action,都会创建一个Action对象、值栈对象、ActionContext对象; 然后把Action对象放入值栈中; 最后再把值栈对象放入request中,传入jsp页面(key: struts.valueStack); 开发者只需要通过ActionContext对象就可以访问struts的其他的关键对象(ActionContext是给开发者用的,便于学习与使用)
    • 问题 : OgnlContext与ValueStack对象的关系?
  • Struts标签 : Struts标签取值,就使用了Ognl表达式语言
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,313评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,369评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,916评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,333评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,425评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,481评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,491评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,268评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,719评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,004评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,179评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,832评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,510评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,153评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,402评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,045评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,071评论 2 352

推荐阅读更多精彩内容