1. 拦截器

1.1 概述

基本概念

Intercetor, 即为拦截器。

1) 在Struts2中,把每一个功能都用一个个的拦截器实现;用户想用struts的哪个功能的时候,可以自由组装使用。

2Struts2中,为了方法用户对拦截器的引用,提供了拦截器栈的定义,里面可以包含多个拦截器。   文件夹(文件, 文件2)  拦截器栈(拦截器,拦截器2)

3Struts2中,如果用户没有指定执行哪些拦截器,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 拦截器案例

需求:

登陆后,显示列表!

案例准备:

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>

 

 

 

 

2. Struts2中的国际化

回顾:Servlet 中国际化:

1. 写资源文件

基础名.properties  【默认的语言环境的配置】  

基础名_语言简称_国家简称.properties

2. 读取资源文件,再使用

程序:ResourceBundle

Jsp:   jstl提供的格式化与国际化标签库。

 

 

 

Struts2中国际化:

1. 写资源文件  (servlet)

2. 读资源文件

程序:ResourceBundle   (servlet)

JSP:  

1jstl表亲啊  (同servlet

2struts标签获取资源文件内容

 

区别:

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:i18n name="cn.itcast.config.msg">

<s:text>  标签必须放到标签体中。

</s:i18n>

 

 

3. Ognl表达式语言

概述

OGNL表达式

       OGNLObject 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)

ü 作用

页面取值用。

 

El表达式语言,用于页面取值,jsp页面取值的标准。(默认直接可以使用)

(应用范围更广。)

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对象

ValueStack, 即值栈对象。

值栈对象:

是整个struts数据存储的核心,或者叫中转站。

用户每次访问strutsaction,都会创建一个Action对象、值栈对象、ActionContext对象;  然后把Action对象放入值栈中; 最后再把值栈对象放入request中,传入jsp页面。

(key: struts.valueStack);     开发者只需要通过ActionContext对象就可以访问struts的其他的关键对象。 (ActionContext是给开发者用的,便于学习与使用。)

posted on 2017-03-03 16:06  夏le夏天  阅读(192)  评论(0)    收藏  举报