zoukankan      html  css  js  c++  java
  • (转)OGNL与值栈

    http://blog.csdn.net/yerenyuan_pku/article/details/67709693

    OGNL的概述

    什么是OGNL

    据度娘所说:

    OGNL是Object-Graph Navigation Language的缩写,它是一种功能强大的表达式语言,通过它简单一致的表达式语法,可以存取对象的任意属性,调用对象的方法,遍历整个对象的结构图,实现字段类型转化功能。它使用相同的表达式去存取对象的属性。

    OGNL的全称是对象图导航语言(Object-Graph Navigation Language),它是一种功能强大的开源表达式语言,使用这种表达式语言,可以通过某种表达式语法,存取Java对象的任意属性,调用Java对象的方法,同时能够自动实现必要的类型转换。如果把表达式看作是一个带有语义的字符串,那么OGNL无疑成为了这个语义字符串与Java对象之间沟通的桥梁。

    OGNL的作用

    Struts2默认的表达式语言就是OGNL,它具有以下特点:

    • 支持对象方法调用。例如:objName.methodName()
    • 支持类静态方法调用和值访问,表达式的格式为@[类全名(包括类路径)]@[方法名 | 值名],例如:@java.lang.String@format(‘foo %s’, ‘bar’)
    • 支持赋值操作和表达式串联,例如:price=100, discount=0.8, calculatePrice(),在方法中进行乘法计算会返回80
    • 访问OGNL上下文(OGNL context)和ActionContext
    • 操作集合对象

    OGNL的要素

    了解什么是OGNL及其特点后,接下来,分析一下OGNL的结构。OGNL的操作实际上就是围绕着OGNL结构的三个要素而进行的,分别是表达式(Expression)、根对象(Root Object)、上下文环境(Context),下面分别讲解这三个要素,具体如下:

    1. 表达式 
      表达式是整个OGNL的核心,OGNL会根据表达式去对象中取值。所有OGNL操作都是针对表达式解析后进行的。它表明了此次OGNL操作要”做什么”。表达式就是一个带有语法含义的字符串,这个字符串规定了操作的类型和操作的内容。OGNL支持大量的表达式语法,不仅支持这种”链式”对象访问路径,还支持在表达式中进行简单的计算。
    2. 根对象(Root) 
      Root对象可以理解为OGNL的操作对象,表达式规定了”做什么”,而Root对象则规定了”对谁操作”。OGNL称为对象导航语言,所谓对象图,即以任意一个对象为根,通过OGNL可以访问与这个对象关联的其他对象。
    3. Context对象 
      实际上OGNL的取值还需要一个上下文环境。设置了Root对象,OGNL可以对Root对象进行取值或写值等操作,Root对象所在环境就是OGNL的上下文环境(Context)。上下文环境规定了OGNL的操作”在哪里进行”。上下文环境Context是一个Map类型的对象,在表达式中访问Context中的对象,需要使用”#”号加上对象名称,即”#对象名称”的形式。

    总结来说:OGNL是一种表达式语言,之前我们学过el表达式,el表达式用于在JSP页面中获取域对象里面的值,但OGNL表达式比el表达式功能更加强大,我们使用OGNL表达式主要做的事情是在Struts2里面获取值栈中的数据(OGNL在Struts2里面经常和Struts2标签一起使用)。这就是我们接下来要重点讲解的内容,即在Struts2里面如何获取值栈ZH数据。 
    我们还需要注意一点:OGNL本身不是Struts2框架的一部分,而是一个单独的项目,只不过它经常和Struts2框架一起使用而已。要使用OGNL表达式,首先要导入Jar包,在Struts2框架里面提供了OGNL表达式的Jar包: 
    这里写图片描述

    OGNL的入门

    前面已经提到过OGNL支持对象方法调用,比如访问对象方法和访问静态方法,这里结合一些案例来演示OGNL是如何调用方法的。 
    案例,使用Struts2的标签+OGNL表达式来计算字符串”hello”的长度,即演示如何调用对象的方法。 
    在Eclipse中创建一个名为struts2_day03的Web项目,之后快速搭建好Struts2框架的开发环境。接着在WebContent目录下新建一个JSP页面——ognl.jsp,内容如下:

    <%@ page language="java" contentType="text/html; charset=UTF-8"
        pageEncoding="UTF-8"%>
    <%@ taglib uri="/struts-tags" prefix="s" %>
    <!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
    <html>
    <head>
    <meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
    <title>Insert title here</title>
    </head>
    <body>
        <!-- 使用OGNL计算字符串的长度 -->
        <!-- 1.Strust2标签,在页面中引入Strust2的标签库,
            使用标签,在标签里面有一个value属性,value属性值写的是OGNL表达式
        -->
        <s:property value="'hello'.length()" />
    </body>
    </html>

    然后在浏览器地址栏中输入URL地址http://localhost:8080/struts2_day03/ognl.jsp,可以看到最终的输出结果为5。 
    通过这个案例可得出两点:

    1. OGNL表达式要和Struts2标签(<s:property>)一起使用。
    2. 在该Struts2标签里面有一个value属性,在value属性值里面要写上OGNL表达式。

    值栈的概述

    什么是值栈

    ValueStack是Struts2的一个接口,字面意义为值栈,OgnlValueStack是ValueStack的实现类,客户端发起一个请求,Struts2框架会创建一个Action实例,同时创建一个OgnlValueStack值栈实例,OgnlValueStack贯穿整个Action的生命周期,Struts2中使用OGNL将请求Action的参数封装为对象存储到值栈中,并通过OGNL表达式读取值栈中的对象属性值。 
    我们之前学习过,比如说在Servlet中要把数据传递到页面中显示,就要在Servlet里面把数据放到域对象里面,然后在JSP页面中使用el表达式获取域对象里面的值。而在Struts2框架里面提供了一个东西——值栈,它类似于域对象,值栈应用在Struts2的Action类里面,我们在值栈中可以存值和取值。 
    接下来我就来讲讲值栈的存储位置,在讲这个知识点之前,我们得清楚Servlet和Action之间的区别:

    • Servlet默认在第一次访问的时候创建,而且只会创建一次,可以理解为它是单实例对象。
    • Action在访问的时候创建,每次访问Action的时候都会创建一个Action对象,可以理解为它是多实例对象。

    我们知道了每次访问Action的时候,都会创建一个Action对象之后,就来讲值栈的存储位置,即在每个Action对象里面都会存在一个值栈对象。那么值栈的使用范围呢?值栈是使用在Action范围的。

    获取值栈对象

    值栈存在于每个Action对象里面。那到底如何获取值栈对象呢?步骤为:

    1. 使用ActionContext类,得到ActionContext对象。
    2. 使用ActionContext对象里面的方法得到值栈对象。
    // 1.获取ActionContext对象
    ActionContext context = ActionContext.getContext();
    // 2.调用ActionContext对象的方法获取到值栈对象
    ValueStack stack1 = context.getValueStack();

    注意:在一个Action对象里面只有一个值栈对象。下面就来举例说明这点。 
    在src目录下创建一个cn.itcast.action包,并在该包下新建一个Action类——UserAction.java。

    public class UserAction extends ActionSupport {
    
        @Override
        public String execute() throws Exception {
            // 1.获取ActionContext对象
            ActionContext context = ActionContext.getContext();
            // 2.调用ActionContext对象的方法获取到值栈对象
            ValueStack stack1 = context.getValueStack();
    
            ValueStack stack2 = context.getValueStack();
    
            System.out.println(stack1 == stack2); // true
    
            return NONE;
        }
    }

    然后在struts.xml核心配置文件中配置好该Action。

    <?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="demo1" extends="struts-default" namespace="/">
            <action name="user" class="cn.itcast.action.UserAction"></action>
        </package>
    </struts>

    最后在浏览器地址栏中输入URL地址http://localhost:8080/struts2_day03/user.action,可以看到Eclipse控制台输出结果为true。这也就说明了在一个Action对象里面只有一个值栈对象。

    值栈的内部结构

    通过查看OgnlValueStack类的源码,可发现在OgnlValueStack类中包含两部分,值栈和map(即OGNL上下文) 

    • Context:即OgnlContext上下文,它是一个Map结构,上下文中存储了一些引用,parameters、request、session、application等,上下文的Root为CompoundRoot。 
      OgnlContext中的一些引用:

      • parameters:该Map中包含当前请求的请求参数
      • request:该Map中包含当前request对象中的所有属性
      • session:该Map中包含当前session对象中的所有属性
      • application:该Map中包含当前application对象中的所有属性
      • attr:该Map按如下顺序来检索某个属性:request、session、application。
    • CompoundRoot:存储了action实例,它作为OgnlContext的Root对象。 
      CompoundRoot继承ArrayList实现压栈和出栈的功能,拥有栈的特点,先进后出,后进先出,最后压进栈的数据在栈顶。我们把它称为对象栈。

    Struts2对原OGNL作出的改进就是Root使用CompoundRoot(自定义栈),使用OgnlValueStack的findValue方法可以在CompoundRoot中从栈顶向栈底找查找的对象的属性值。CompoundRoot作为OgnlContext的Root对象,并且在CompoundRoot中action实例位于栈顶,当读取action的属性值时会先从栈顶对象中找对应的属性,如果找不到则继续找栈中的其他对象,如果找到则停止查找。

    现在通过一个案例来分析一下值栈的内部结构。在UserAction类代码的如下位置加上一个断点:

    ValueStack stack1 = context.getValueStack();

    然后以断点模式来启动tomcat,Watch一下stack1值,可得到如下结果: 

    其实,在Struts2的标签里面有一个标签可以看到值栈的内部结构,这个标签就是<s:debug>。现在我们就来举例演示。 
    在src目录下新建一个cn.itcast.valuestack包,并在该包下创建一个Action类——ValueStackDemo1.java。

    public class ValueStackDemo1 extends ActionSupport {
        @Override
        public String execute() throws Exception {
            return "demo1";
        }
    }
    • 1

    接着在struts.xml核心配置文件中配置好该Action类,如下:

    <?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="demo1" extends="struts-default" namespace="/">
            <action name="user" class="cn.itcast.action.UserAction"></action>
            <action name="stackdemo1" class="cn.itcast.valuestack.ValueStackDemo1">
                <result name="demo1">/demo1.jsp</result>
            </action>
        </package>
    </struts>

    紧接着在WebContent目录下新建一个JSP页面——demo1.jsp。

    <%@ page language="java" contentType="text/html; charset=UTF-8"
        pageEncoding="UTF-8"%>
    <%@ taglib uri="/struts-tags" prefix="s" %>
    <!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
    <html>
    <head>
    <meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
    <title>Insert title here</title>
    </head>
    <body>
        <!-- 使用这个标签可以看到值栈内部结构 -->
        <s:debug></s:debug>
    </body>
    </html>

    最后在浏览器地址栏中输入URL地址http://localhost:8080/struts2_day03/stackdemo1.action,可以看到如下界面: 

    点击这个超链接即可看到值栈结构,如下: 

    注意:<s:debug>这个标签只是在调试时候使用。

    向值栈中放入数据

    向值栈中放入数据有三种方式,分别是:

    1. 第一种方式: 获取值栈对象,调用值栈对象里面的set方法。
    2. 第二种方式:获取值栈对象,调用值栈对象里面的push方法。
    3. 第三种方式:在action成员变量位置定义变量,生成这个变量的get方法。

    不过在实际开发中,我们一般都使用的是第三种方式。所有这种方式,我们应该重点掌握。

    向值栈中放入字符串

    下面我们分别用三种方式来演示如何向值栈中放入字符串。

    第一种方式: 获取值栈对象,调用值栈对象里面的set方法

    在src目录下新建一个cn.itcast.valuestack包,并在该包下创建一个Action类——ValueStackDemo1.java。

    public class ValueStackDemo1 extends ActionSupport {
    
        @Override
        public String execute() throws Exception {
    
            // 第一种方式
            // 1.获取值栈对象
            ActionContext context = ActionContext.getContext();
            ValueStack stack = context.getValueStack();
            // 2.调用值栈对象里面的set方法
            stack.set("username", "liayun");
    
            return "demo1";
        }
    
    }

    然后在struts.xml核心配置文件中配置好该Action类。

    <?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="demo1" extends="struts-default" namespace="/">
            <action name="user" class="cn.itcast.action.UserAction"></action>
            <action name="stackdemo1" class="cn.itcast.valuestack.ValueStackDemo1">
                <result name="demo1">/demo1.jsp</result>
            </action>
        </package>
    </struts>

    demo1.jsp页面内容不必修改,最后在浏览器地址栏中输入URL地址http://localhost:8080/struts2_day03/stackdemo1.action,点击超链接即可看到值栈结构,如下: 

    第二种方式:获取值栈对象,调用值栈对象里面的push方法

    修改Action类——ValueStackDemo1.java的代码为:

    public class ValueStackDemo1 extends ActionSupport {
    
        @Override
        public String execute() throws Exception {
            // 向变量里面设置值
            // username = "叶十一少";
    
            // 第一种方式
            // 1.获取值栈对象
            ActionContext context = ActionContext.getContext();
            ValueStack stack = context.getValueStack();
            // 2.调用值栈对象里面的set方法
            stack.set("username", "liayun");
    
            // 调用push方法
            stack.push("caonima");
    
            return "demo1";
        }
    
    }

    struts.xml核心配置文件和demo1.jsp页面内容都不必修改,然后在浏览器地址栏中输入URL地址http://localhost:8080/struts2_day03/stackdemo1.action,点击超链接即可看到值栈结构,如下: 

    第三种方式:在action成员变量位置定义变量,生成这个变量的get方法

    修改Action类——ValueStackDemo1.java的代码为:

    public class ValueStackDemo1 extends ActionSupport {
    
        private String username;
    
        public String getUsername() {
            return username;
        }
    
        @Override
        public String execute() throws Exception {
            // 向变量里面设置值
            username = "叶十一少";
    
            return "demo1";
        }
    }

    struts.xml核心配置文件和demo1.jsp页面内容也都不必修改,然后在浏览器地址栏中输入URL地址http://localhost:8080/struts2_day03/stackdemo1.action,点击超链接即可看到值栈结构,如下: 

    向值栈中放入对象

    我举例来演示如何向值栈中放入对象。 
    首先在cn.itcast.valuestack包中创建一个JavaBean——User.java。

    public class User {
    
        private String username;
        private String address;
        public String getUsername() {
            return username;
        }
        public void setUsername(String username) {
            this.username = username;
        }
        public String getAddress() {
            return address;
        }
        public void setAddress(String address) {
            this.address = address;
        }
    
    }

    然后再在该包下编写一个Action类——ObjectValueStack.java。

    public class ObjectValueStack extends ActionSupport {
    
        // 1.声明对象的变量
        private User user = new User();
    
        // 2.生成b变量的get方法
        public User getUser() {
            return user;
        }
    
        public String execute() throws Exception {
            // 3.向值栈的对象里面设置值(即向user变量里面设置值)
            user.setUsername("凤姐");
            user.setAddress("美国");
    
            return "objectValue";
        }
    
    }

    接着在struts.xml核心配置文件中配置好该Action类。

    <?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="demo1" extends="struts-default" namespace="/">
            <action name="user" class="cn.itcast.action.UserAction"></action>
            <action name="stackdemo1" class="cn.itcast.valuestack.ValueStackDemo1">
                <result name="demo1">/demo1.jsp</result>
            </action>
            <action name="objectstack" class="cn.itcast.valuestack.ObjectValueStack">
                <result name="objectValue">/demo1.jsp</result>
            </action>
        </package>
    </struts>

    demo1.jsp页面的内容不必修改,最后在浏览器地址栏中输入URL地址http://localhost:8080/struts2_day03/objectstack.action,点击超链接即可看到值栈结构,如下: 

    向值栈中放入List集合

    我举例来演示如何向值栈中放入List集合。 
    首先在cn.itcast.valuestack包中创建一个Action类——ListValueStack.java。

    public class ListValueStack extends ActionSupport {
    
        // 1.声明list集合的变量
        private List<User> list = new ArrayList<User>();
    
        // 2.生成get方法
        public List<User> getList() {
            return list;
        }
    
        @Override
        public String execute() throws Exception {
            // 3.向值栈的list里面设置值
            User u1 = new User();
            u1.setUsername("如花");
            u1.setAddress("越南");
    
            User u2 = new User();
            u2.setUsername("小金");
            u2.setAddress("朝鲜");
    
            list.add(u1);
            list.add(u2);
            return "list";
        }
    
    }

    接着在struts.xml核心配置文件中配置好该Action类。

    <?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="demo1" extends="struts-default" namespace="/">
            <action name="user" class="cn.itcast.action.UserAction"></action>
            <action name="stackdemo1" class="cn.itcast.valuestack.ValueStackDemo1">
                <result name="demo1">/demo1.jsp</result>
            </action>
            <action name="objectstack" class="cn.itcast.valuestack.ObjectValueStack">
                <result name="objectValue">/demo1.jsp</result>
            </action>
            <action name="liststack" class="cn.itcast.valuestack.ListValueStack">
                <result name="list">/demo1.jsp</result>
            </action>
        </package>
    </struts>

    demo1.jsp页面的内容不必修改,最后在浏览器地址栏中输入URL地址http://localhost:8080/struts2_day03/liststack.action,点击超链接即可看到值栈结构,如下: 

    从值栈中获取数据

    在jsp页面中,可使用Struts2标签+OGNL表达式获取到值栈中的数据。类似于:

    <s:property value="ognl表达式" />

    获取字符串

    上面我们已把字符串放到值栈里面,如:

    public class ValueStackDemo1 extends ActionSupport {
    
        private String username;
    
        public String getUsername() {
            return username;
        }
    
        @Override
        public String execute() throws Exception {
            // 向变量里面设置值
            username = "叶十一少";
    
            return "demo1";
        }
    }

    那么我们可以在jsp页面中获取值栈里面的字符串的值,如下:

    <!-- 获取值栈里面的字符串的值 -->
    <s:property value="username" />

    获取对象

    上面我们已把对象放到值栈里面了,如:

    public class ObjectValueStack extends ActionSupport {
    
        // 1.声明对象的变量
        private User user = new User();
    
        // 2.生成b变量的get方法
        public User getUser() {
            return user;
        }
    
        public String execute() throws Exception {
            // 3.向值栈的对象里面设置值(即向user变量里面设置值)
            user.setUsername("凤姐");
            user.setAddress("美国");
    
            return "objectValue";
        }
    
    }

    那么我们可以在jsp页面中获取值栈里面的对象的值,如下:

    <!-- 获取值栈的对象的值 -->
    <s:property value="user" /><br/> <!-- 得到是一个User对象 -->
    <s:property value="user.username" /><br/>
    <s:property value="user.address" />

    获取List集合

    上面我们已把List集合放到值栈里面了,如:

    public class ListValueStack extends ActionSupport {
    
        // 1.声明list集合的变量
        private List<User> list = new ArrayList<User>();
    
        // 2.生成get方法
        public List<User> getList() {
            return list;
        }
    
        @Override
        public String execute() throws Exception {
            // 3.向值栈的list里面设置值
            User u1 = new User();
            u1.setUsername("如花");
            u1.setAddress("越南");
    
            User u2 = new User();
            u2.setUsername("小金");
            u2.setAddress("朝鲜");
    
            list.add(u1);
            list.add(u2);
            return "list";
        }
    
    }

    那么在jsp页面中我们如何获取值栈里面的List集合的数据呢?共有三种方式,下面我们分别来加以说明。 
    第一种方式比较傻逼,根本就不会使用到。

    <!-- 获取值栈里面的List集合的数据 -->
    <!-- 第一种方式 -->
    <s:property value="list[0].username" />
    <s:property value="list[0].address" /><br/>
    <s:property value="list[1].username" />
    <s:property value="list[1].address" /><br/>

    第二种方式——在Struts2标签里面有一个可以进行遍历操作的标签,类似于JSTL标签库中的foreach标签,第二种方式就要使用到它。

    <!-- 第二种方式 -->
    <s:iterator value="list">
        <!-- 遍历得到list集合里面的每个User对象 -->
        <s:property value="username" />
        ------
        <s:property value="address" />
        <br/>
    </s:iterator>

    第三种方式:

    <!-- 第三种方式 -->
    <s:iterator value="list" var="user">
        <!-- 遍历得到list集合里面的每个User对象 -->
        <s:property value="#user.username" />
        ======
        <s:property value="#user.address" />
        <br/>
    </s:iterator>

    因为我们现在操作的数据都是在值栈的root里面,如果你使用Struts2里面的iterator标签的时候,在标签里面写了var,它会把list集合中每次遍历出来的User对象放到context里面。而在获取context里面的值,写OGNL表达式时,要求在OGNL表达式之前添加#号!

    获取值栈数据的其他操作

    之前,我们已学会了使用值栈对象的set方法把数据放到值栈里面,也学会了使用值栈对象里面的push方法把数据放到值栈里面。如:

    public class ValueStackDemo1 extends ActionSupport {
    
        @Override
        public String execute() throws Exception {
            // 向变量里面设置值
            // username = "叶十一少";
    
            // 第一种方式
            // 1.获取值栈对象
            ActionContext context = ActionContext.getContext();
            ValueStack stack = context.getValueStack();
            // 2.调用值栈对象里面的set方法
            stack.set("username", "liayun");
    
            // 调用push方法
            stack.push("caonima");
    
            return "demo1";
        }
    
    }

    那么我们可以在jsp页面中获取值栈里面使用set方法和push方法设置的值 ,如下:

    <!-- 获取使用set方法设置的值 -->
    <s:property value="username" />
    <!-- 
        获取使用push方法设置的值
        把你放的这些值会存到默认的一个数组里面去,数组的名称是top
    -->
    <s:property value="[0].top" /> <!-- 得到栈顶的数据 -->

    这种方式只要求理解即可,不必深究,因为没什么屌用!

    EL表达式为什么能获取值栈数据?

    当然了,我们使用EL表达式是能够访问值栈的,但我们应该明确一点,EL表达式是用于在jsp中获取域对象里面的数据的,其实EL表达式不能直接获取值栈数据。然而我们也说了,EL表达式是能获取值栈数据的,这里面的底层原因我们肯定是要知道的,归根结底就是要阅读源码: 
    Struts2底层对Request对象的getAttribute()方法进行了增强,可通过查看StrutsRequestWrapper类的源码得知:

    public class StrutsRequestWrapper extends HttpServletRequestWrapper {
    
        private static final String REQUEST_WRAPPER_GET_ATTRIBUTE = "__requestWrapper.getAttribute";
        private final boolean disableRequestAttributeValueStackLookup;
    
        /**
         * The constructor
         * @param req The request
         */
        public StrutsRequestWrapper(HttpServletRequest req) {
            this(req, false);
        }
    
        /**
         * The constructor
         * @param req The request
         * @param disableRequestAttributeValueStackLookup flag for disabling request attribute value stack lookup (JSTL accessibility)
         */
        public StrutsRequestWrapper(HttpServletRequest req, boolean disableRequestAttributeValueStackLookup) {
            super(req);
            this.disableRequestAttributeValueStackLookup = disableRequestAttributeValueStackLookup;
        }
    
        /**
         * Gets the object, looking in the value stack if not found
         *
         * @param key The attribute key
         */
        public Object getAttribute(String key) {
            if (key == null) {
                throw new NullPointerException("You must specify a key value");
            }
    
            if (disableRequestAttributeValueStackLookup || key.startsWith("javax.servlet")) {
                // don't bother with the standard javax.servlet attributes, we can short-circuit this
                // see WW-953 and the forums post linked in that issue for more info
                return super.getAttribute(key);
            }
    
            ActionContext ctx = ActionContext.getContext();
            Object attribute = super.getAttribute(key);
    
            if (ctx != null && attribute == null) {
                boolean alreadyIn = isTrue((Boolean) ctx.get(REQUEST_WRAPPER_GET_ATTRIBUTE));
    
                // note: we don't let # come through or else a request for
                // #attr.foo or #request.foo could cause an endless loop
                if (!alreadyIn && !key.contains("#")) {
                    try {
                        // If not found, then try the ValueStack
                        ctx.put(REQUEST_WRAPPER_GET_ATTRIBUTE, Boolean.TRUE);
                        ValueStack stack = ctx.getValueStack();
                        if (stack != null) {
                            attribute = stack.findValue(key);
                        }
                    } finally {
                        ctx.put(REQUEST_WRAPPER_GET_ATTRIBUTE, Boolean.FALSE);
                    }
                }
            }
            return attribute;
        }
    }

    getAttribute()是用于获取域对象里面的值的方法。它的一个大致上的执行流程是:

    1. 首先到request里面找是否有值,如果request域里面有值,直接返回。
    2. 如果域对象里面没有值,则得到值栈对象,从值栈对象里面把值获取到,最后放到域对象里面去。

    EL的特殊字符的使用

    #号的使用

    之前已说过#号是用于获取context里面的数据的。context是一个key-value键值对的Map集合,里面有request对象的引用。现在我就来举例说明#号的使用。 
    首先在cn.itcast.valuestack包中编写一个Action类——ContextValueAction.java。

    public class ContextValueAction extends ActionSupport {
        @Override
        public String execute() throws Exception {
            // 向request域里面放值
            HttpServletRequest request = ServletActionContext.getRequest();
            request.setAttribute("username", "requestValue");
    
            return "contextvalue";
        }
    }

    并在struts.xml核心配置文件中配置好该Action类。

    <action name="context" class="cn.itcast.valuestack.ContextValueAction">
        <result name="contextvalue">/contextvalue.jsp</result>
    </action>

    最后在WebContent目录下新建一个jsp页面——contextvalue.jsp。

    <%@ page language="java" contentType="text/html; charset=UTF-8"
        pageEncoding="UTF-8"%>
    <%@ taglib uri="/struts-tags" prefix="s" %>
    <!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
    <html>
    <head>
    <meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
    <title>Insert title here</title>
    </head>
    <body>  
        <!-- 获取context里面的数据 -->
        <s:property value="#request.username" />
    </body>
    </html>

    这样即可获取到request域里面的数据。

    %的使用

    %要和Struts2的表单标签一起使用,至于Struts2的表单标签,后面我会介绍到。 
    这儿是紧接着上面的案例来讲的,我使用Struts2的表单标签,并使用OGNL表达式来显示值,如下;

    <%@ page language="java" contentType="text/html; charset=UTF-8"
        pageEncoding="UTF-8"%>
    <%@ taglib uri="/struts-tags" prefix="s" %>
    <!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
    <html>
    <head>
    <meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
    <title>Insert title here</title>
    </head>
    <body>  
        <s:textfield value="#request.username"></s:textfield>
    </body>
    </html>

    在浏览器地址栏中输入URL地址http://localhost:8080/struts2_day03/context.action,可看到如下界面: 

    发现OGNL表达式作为字符串显示出来了,而没有作为OGNL表达式来执行。要解决这个问题,就要使用%来强制解析OGNL表达式(即使用%让表单标签里面的值作为OGNL表达式执行)

    <%@ page language="java" contentType="text/html; charset=UTF-8"
        pageEncoding="UTF-8"%>
    <%@ taglib uri="/struts-tags" prefix="s" %>
    <!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
    <html>
    <head>
    <meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
    <title>Insert title here</title>
    </head>
    <body>  
        <s:textfield value="%{#request.username}"></s:textfield>
    </body>
    </html>

    这时,才可看到正确的值。

    $的使用

    我们可在配置文件中使用OGNL表达式,即可在Struts2的配置文件中使用.xml文件或者是属性文件。关于 $符号的使用,后面会简单介绍到。

  • 相关阅读:
    动态规划例题
    c++ 进制转换函数
    约瑟夫问题
    set的基本使用
    stl中的二分查找
    1.生成的接口返回参数不包括系统自带的参数
    查看被锁定的表,并解锁
    添加、修改表中的字段
    NPOI简单示例2—合并表头
    NPOI简单示例
  • 原文地址:https://www.cnblogs.com/telwanggs/p/6970546.html
Copyright © 2011-2022 走看看