struts2 之 OGNL
要谈OGNL在Struts2中的应用,首先得明白OGNL到底是什么
在值栈中的对象,可以直接访问,但是用#.对象的时候却又访问不到,但用#this.对象的时候又可以访问得到,用%所有的没关系的
OGNL 的历史
OGNL 最初是为了能够使用对象的属性名来建立 UI 组件 (component) 和 控制器 (controllers) 之间的联系,简单来说就是:视图 与 控制器 之间数据的联系。后来为了应付更加复杂的数据关系,Drew Davidson 发明了一个被他称为 KVCL(Key-Value Coding Language) 的语言。 Luke 参与进来后,用 ANTLR 来实现了该语言,并给它取了这个新名字,他后来又使用 JavaCC 重新实现了该语言。目前 OGNL 由 Drew 来负责维护。目前很多项目中都用到了 OGNL,其中不乏为大家所熟知的,例如几个流行的 web 应用框架:WebWork【当然struts2也可以说是WebWork升级版】,Tapestry 等。
什么是 OGNL?
OGNL 是 Object-Graph Navigation Language 的缩写,从语言角度来说:它是一个功能强大的表达式语言,用来获取和设置 java 对象的属性 , 它旨在提供一个更高抽象度语法来对 java 对象图进行导航,OGNL 在许多的地方都有应用,例如:
1)作为 GUI 元素(textfield,combobox, 等)到模型对象的绑定语言。
2)数据库表到 Swing 的 TableModel 的数据源语言。
3)web 组件和后台 Model 对象的绑定语言 (WebOGNL,Tapestry,WebWork,WebObjects) 。
4)作为 Jakarata Commons BeanUtils 或者 JSTL 的表达式语言的一个更具表达力的替代语言。
另外,java 中很多可以做的事情,也可以使用 OGNL 来完成,例如:列表映射和选择。对于开发者来说,使用 OGNL,可以用简洁的语法来完成对 java 对象的导航。通常来说:通过一个“路径”来完成对象信息的导航,这个“路径”可以是到 java bean 的某个属性,或者集合中的某个索引的对象,等等,而不是直接使用 get 或者 set 方法来完成
OGNL 的基本语法
OGNL 表达式一般都很简单。虽然 OGNL 语言本身已经变得更加丰富了也更强大了,但是一般来说那些比较复杂的语言特性并未影响到 OGNL 的简洁:简单的部分还是依然那么简单。比如要获取一个对象的 name 属性,OGNL 表达式就是 name, 要获取一个对象的 headline 属性的 text 属性,OGNL 表达式就是 headline.text 。 OGNL 表达式的基本单位是“导航链”,往往简称为“链”。最简单的链包含如下部分:
表达式组成部分 示例
属性名称 如上述示例中的 name 和 headline.text
方法调用 hashCode() 返回当前对象的哈希码。
数组元素 listeners[0] 返回当前对象的监听器列表中的第一个元素。
所有的 OGNL 表达式都基于当前对象的上下文来完成求值运算,链的前面部分的结果将作为后面求值的上下文。你的链可以写得很长,例如:
name.toCharArray()[0].numericValue.toString()
上面的表达式的求值步骤:
提取根 (root) 对象的 name 属性。
调用上一步返回的结果字符串的 toCharArray() 方法。
提取返回的结果数组的第一个字符。
获取字符的 numericValue 属性,该字符是一个 Character 对象,Character 类有一个 getNumericValue() 方法。
调用结果 Integer 对象的 toString() 方法。
上面的例子只是用来得到一个对象的值,OGNL 也可以用来去设置对象的值。当把上面的表达式传入 Ognl.setValue() 方法将导致 InappropriateExpressionException,因为链的最后的部分(toString())既不是一个属性的名字也不是数组的某个元素。了解了上面的语法基本上可以完成绝大部分工作了。
OGNL 表达式
1)常量:字符串:“ hello ” 字符:‘ h ’ 数字:除了像 java 的内置类型 int,long,float 和 double,Ognl 还有如例:
0.01B,相当于 java.math.BigDecimal,使用’ b ’或者’ B ’后缀。 100000H,相当于 java.math.BigInteger,使用’ h ’ 或 ’ H ’ 后缀。
2)属性的引用例如:user.name
3)变量的引用例如:#name
4)静态变量的访问使用 @class@field
5)静态方法的调用使用 @class@method(args), 如果没有指定 class 那么默认就使用 java.lang.Math.
6)构造函数的调用例如:new java.util.ArrayList();
其它的 Ognl 的表达式可以参考 Ognl 的语言手册。
OGNL的性能
OGNL,或者说表达式语言的性能主要又两方面来决定,一个就是对表达式的解析 (Parser),另一个是表达式的执行,OGNL 采用 javaCC 来完成 parser 的实现,在 OGNL 2.7 中又对 OGNL 的执行部分进行了加强,使用 javasisit 来 JIT(Just-In-Time) 的生成 byte code 来完成表达式的执行。 Ognl 给这个功能的名字是:OGNL Expression Compilation 。基本的使用方法是:
SimpleObject root = new SimpleObject();
OgnlContext context = (OgnlContext) Ognl.createDefaultContext(null);
Node node = (Node) Ognl.compileExpression(context, root, "user.name");
String userName = (String)node.getAccessor().get(context, root);
表达式语言主要有以下几大好处:
1)避免(MyType) request.getAttribute()和myBean.getMyProperty()之类的语句,使页面更简洁;
2)支持运算符(如+-*/),比普通的标志具有更高的自由度和更强的功能;
3)简单明了地表达代码逻辑,使用代码更可读与便于维护。
Struts 2中的表达式语言
Struts 2支持以下几种表达式语言:
1)OGNL(Object-Graph Navigation Language),可以方便地操作对象属性的开源表达式语言;
2)JSTL(JSP Standard Tag Library),JSP 2.0集成的标准的表达式语言;
3)Groovy,基于Java平台的动态语言,它具有时下比较流行的动态语言(如Python、Ruby和Smarttalk等)的一些起特性;
4)Velocity,严格来说不是表达式语言,它是一种基于Java的模板匹配引擎,具说其性能要比JSP好。
Struts 2默认的表达式语言是OGNL,原因是它相对其它表达式语言具有下面几大优势:
支持对象方法调用,如xxx.doSomeSpecial();
支持类静态的方法调用和值访问,表达式的格式为@[类全名(包括包路径)]@[方法名 | 值名],例如:@java.lang.String@format('foo %s', 'bar')或@tutorial.MyConstant@APP_NAME;
支持赋值操作和表达式串联,如price=100, discount=0.8, calculatePrice(),这个表达式会返回80;
访问OGNL上下文(OGNL context)和ActionContext;
操作集合对象。
Struts2中ONGL的使用示例
index.html
1 <span style="font-size:18px;"><span style="font-size:18px;"><html>
2 <head>
3 <meta http-equiv="Content-Type" content="text/html; charset=UTF-8" />
4 <meta http-equiv="refresh" content="0; url=ognl.action" />
5 </head>
6 </html></span></span>
User.java
1 import java.util.Date;
2
3
4 public class User {
5 private Integer id;
6 private String loginname;
7 private Double score;
8 private Boolean gender;
9 private Character cha;
10 private Date birthday;
11
12 public User(){}
13
14 public User(Integer id, String loginname, Double score, Boolean gender,
15 Character cha, Date birthday) {
16 this.id = id;
17 this.loginname = loginname;
18 this.score = score;
19 this.gender = gender;
20 this.cha = cha;
21 this.birthday = birthday;
22 }
23 public Integer getId() {
24 return id;
25 }
26 public void setId(Integer id) {
27 this.id = id;
28 }
29 public String getLoginname() {
30 return loginname;
31 }
32 public void setLoginname(String loginname) {
33 this.loginname = loginname;
34 }
35 public Double getScore() {
36 return score;
37 }
38 public void setScore(Double score) {
39 this.score = score;
40 }
41 public Boolean getGender() {
42 return gender;
43 }
44 public void setGender(Boolean gender) {
45 this.gender = gender;
46 }
47 public Character getCha() {
48 return cha;
49 }
50 public void setCha(Character cha) {
51 this.cha = cha;
52 }
53 public Date getBirthday() {
54 return birthday;
55 }
56 public void setBirthday(Date birthday) {
57 this.birthday = birthday;
58 }
59
60 public String info() {
61 return "User [birthday=" + birthday + ", cha=" + cha + ", gender="
62 + gender + ", id=" + id + ", loginname=" + loginname
63 + ", score=" + score + "]";
64 }
OGNLAction.java
1 import java.util.ArrayList;
2 import java.util.Date;
3 import java.util.HashMap;
4 import java.util.LinkedHashSet;
5 import java.util.List;
6 import java.util.Map;
7 import java.util.Set;
8
9 import org.apache.struts2.ServletActionContext;
10
11 import com.opensymphony.xwork2.ActionContext;
12 import com.opensymphony.xwork2.ActionSupport;
13
14 public class OGNLAction extends ActionSupport {
15 private static final long serialVersionUID = -2554018432709689579L;
16 private String loginname;
17 private String pwd;
18 private User user;
19 private Set<String> courseSet;
20 private List<String> list;
21 private Map<String,String> map;
22 private List<User> userList;
23
24
25 public String <span style="color:#000066;"><strong>execute()</strong></span> throws Exception{
26 this.loginname = "xkkkkkkkkkkkkkkkkkkkkkkkk";
27 this.user = new User(123, "wrr", 88.9, true, 'B', new Date());
28 this.courseSet = new LinkedHashSet<String>();
29 this.courseSet.add("corejava");
30 this.courseSet.add("JSP/Servlet");
31 this.courseSet.add("S2SH");
32
33 this.list = new ArrayList<String>(this.courseSet);
34 this.map = new HashMap<String, String>();
35 this.map.put("x", "xxx");
36 this.map.put("y", "yyy");
37 this.map.put("z", "zzz");
38
39
40 <span style="color:#993399;">ActionContext context = ActionContext.getContext();
41 context.put("uname", "cheney");
42 context.put("inte", Integer.valueOf(888888));
43 context.put("user2", new User(123, "xxk", 88.9, true, 'B', new Date()));</span>
44
45
46 this.userList = new ArrayList<User>();
47 this.userList.add(new User(1, "zs", 48.9, true, 'D', new Date()));
48 this.userList.add(new User(2, "ls", 68.1, true, 'C', new Date()));
49 this.userList.add(new User(3, "ww", 78.2, false, 'B', new Date()));
50 this.userList.add(new User(4, "zl", 88.3, true, 'A', new Date()));
51
52 //-----------------------------------------------------------------
53 //推荐方式:不会跟Servlet API耦合
54 context.put("reqAtt", "往ActionContext中put的属性");
55 context.getSession().put("sesAtt", "往ActionContext.getSession()中put的属性");
56 context.getApplication().put("appAtt", "往ActionContext.getApplication()中put的属性");
57
58 ServletActionContext.getRequest().setAttribute("reqAtt2", "Request作用域中的属性");
59 ServletActionContext.getRequest().getSession().setAttribute("sesAtt2", "Session作用域中的属性");
60 ServletActionContext.getServletContext().setAttribute("appAtt2", "Application作用域中的属性");
61
62 return SUCCESS;
63 }
64
65
66 public String getAppName(){
67 return "这是OGNL的使用示例代码";
68 }
69
70 public String getLoginname() {
71 return loginname;
72 }
73 public void setLoginname(String loginname) {
74 this.loginname = loginname;
75 }
76 public String getPwd() {
77 return pwd;
78 }
79 public void setPwd(String pwd) {
80 this.pwd = pwd;
81 }
82
83 public User getUser() {
84 return user;
85 }
86
87 public void setUser(User user) {
88 this.user = user;
89 }
90
91 public Set<String> getCourseSet() {
92 return courseSet;
93 }
94
95 public void setCourseSet(Set<String> courseSet) {
96 this.courseSet = courseSet;
97 }
98
99 public List<String> getList() {
100 return list;
101 }
102
103 public void setList(List<String> list) {
104 this.list = list;
105 }
106
107 public Map<String, String> getMap() {
108 return map;
109 }
110
111 public void setMap(Map<String, String> map) {
112 this.map = map;
113 }
114
115 public List<User> getUserList() {
116 return userList;
117 }
118
119 public void setUserList(List<User> userList) {
120 this.userList = userList;
121 }
122
123 }
src/struts.xm
1 <?xml version="1.0" encoding="UTF-8" ?>
2 <!DOCTYPE struts PUBLIC
3 "-//Apache Software Foundation//DTD Struts Configuration 2.1.7//EN"
4 "http://struts.apache.org/dtds/struts-2.1.7.dtd">
5
6 <struts>
7 <!-- 请求参数的编码方式 -->
8 <constant name="struts.i18n.encoding" value="UTF-8"/>
9 <!-- 指定被struts2处理的请求后缀类型。多个用逗号隔开 -->
10 <constant name="struts.action.extension" value="action,do,go,xkk"/>
11 <!-- 当struts.xml改动后,是否重新加载。默认值为false(生产环境下使用),开发阶段最好打开 -->
12 <constant name="struts.configuration.xml.reload" value="true"/>
13 <!-- 是否使用struts的开发模式。开发模式会有更多的调试信息。默认值为false(生产环境下使用),开发阶段最好打开 -->
14 <constant name="struts.devMode" value="false"/>
15 <!-- 设置浏览器是否缓存静态内容。默认值为true(生产环境下使用),开发阶段最好关闭 -->
16 <constant name="struts.serve.static.browserCache" value="false" />
17 <!-- 是否允许在OGNL表达式中调用静态方法,默认值为false -->
18 <constant name="struts.ognl.allowStaticMethodAccess" value="true"/>
19
20 <!-- 指定由spring负责action对象的创建
21 <constant name="struts.objectFactory" value="spring" />
22 -->
23 <!-- 是否开启动态方法调用 -->
24 <constant name="struts.enable.DynamicMethodInvocation" value="false"/>
25
26 <package name="my" extends="struts-default" namespace="/">
27 <action name="ognl" class="com.javacrazyer.web.action.OGNLAction">
28 <result>/ognl_info.jsp</result>
29 </action>
30 </package>
31
32 </struts>
测试:
1 <%@ page language="java" import="java.util.*" pageEncoding="utf-8"%>
2 <%@ taglib uri="/struts-tags" prefix="s" %>
3 <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">
4 <html>
5 <head>
6 <title>OGNL的使用</title>
7 </head>
8 <body>
9 <h3>OGNL的使用</h3><hr/>
10 <span style="color:#6600cc;">访问Action中的普通属性</span>: <s:property value="loginname"/><br/>
11 <span style="color:#ff6600;">访问Action中的对象属性</span>: <s:property value="user.birthday"/><br/>
12 访问Action中的Set属性: <s:property value="courseSet.toArray()[0]"/><br/>
13 访问Action中的List属性: <s:property value="list[0]"/><br/>
14 访问Action中的Map属性的键: <s:property value="map.keys.toArray()[1]"/><br/>
15 访问Action中的Map属性的值: <s:property value="map.values.toArray()[1]"/><br/>
16 访问Action中的Map属性的指定键对应的值: <s:property value="map['z']"/><br/>
17 访问Action中的Map属性的大小: <s:property value="map.size"/><br/>
18 <hr/>
19 访问ActionContext中的普通属性:<s:property value="#inte"/><br/>
20 访问ActionContext中的对象属性:<s:property value="#user2.loginname"/><br/>
21 <hr/>
22 <strong><span style="color:#6600cc;">访问Action中的普通方法:<s:property value="getAppName()"/><br/></span></strong>
23 访问ActionContext中的某个对象上的普通方法:<s:property value="#user2.info()"/><br/>
24 <hr/>
25 <span style="color:#33cc00;">访问静态属性:<s:property value="@java.lang.Math@PI"/><br/>
26 访问静态方法:<s:property value="@java.lang.Math@floor(44.56)"/><br/></span>
27 访问Math类中的静态方法:<s:property value="@@floor(44.56)"/><br/>
28 <hr/>
29 <span style="color:#cc0000;">调用java.util.Date的构造方法</span>:<s:date name="new java.util.Date()" format="yyyy-MM-dd HH:mm:ss"/><br/>
30 调用java.util.Date的构造方法创建对象,再调用它的方法:<s:property value="new java.util.Date().getTime()"/><br/>
31 <hr/>
32 投影查询:获取userList中所有loginname的列表:<s:property value="userList.{loginname}"/><br/>
33 选择查询:获取userList中所有score大于60的loginname列表:<s:property value="userList.{?#this.score>60.0}.{loginname}"/><br/>
34 选择查询:获取userList中所有score大于60并且gender为true的loginname列表:<s:property value="userList.{?(#this.score>60.0 && #this.gender)}.{loginname}"/><br/>
35 选择查询:获取userList中所有score大于60并且gender为true的第一个元素的loginname:<s:property value="userList.{^(#this.score>60.0 && #this.gender)}.{loginname}"/><br/>
36 选择查询:获取userList中所有score大于60并且gender为true的最后一个元素的loginname:<s:property value="userList.{$(#this.score>60.0 && #this.gender)}.{loginname}"/><br/>
37 <hr/>
38 访问名为xxx的请求参数对应的第一个值:<s:property value="#parameters.xxx[0]"/><br/>
39 访问通过ActionContext中放入Request中的属性:<s:property value="#request.reqAtt"/><br/>
40 访问通过ServletActionContext中放入Request中的属性:<s:property value="#request.reqAtt2"/><br/>
41
42 访问通过ActionContext中放入Session中的属性:<s:property value="#session.sesAtt"/><br/>
43 访问通过ServletActionContext中放入Session中的属性:<s:property value="#session.sesAtt2"/><br/>
44 访问通过ActionContext中放入ServletContext中的属性:<s:property value="#application.appAtt"/><br/>
45 访问通过ServletActionContext中放入ServletContext中的属性:<s:property value="#application.appAtt2"/><br/>
46
47 直接访问属性域中指定名称的属性对应的值:<s:property value="#attr.sesAtt2"/><br/>
48 <br/><br/><hr/>
49 <s:iterator value="userList" status="vs">
50 <s:if test="%{#vs.odd}">
51 <span style="color: red">
52 <s:property value="#vs.count"/>: <s:property value="loginname"/>,<s:date name="birthday" format="yyyy-MM-dd HH:mm:ss"/><br/>
53 </span>
54 </s:if>
55 <s:else>
56 <span style="color: blue">
57 <s:property value="#vs.count"/>: <s:property value="loginname"/>,<s:date name="birthday" format="yyyy-MM-dd HH:mm:ss"/><br/>
58 </span>
59 </s:else>
60 </s:iterator>
61
62 <hr/><s:debug/>
63 </body>
64 </html>
总结:
在上边大家都好奇为什么都用struts的S标签,因为OGNL是通常要结合Struts 2的标志一起使用,如<s:property value="xx" />等
Action类与JSP页面之间的数据传递
1) 通过HttpServletRequest,HttpSession,ServletContext来传递数据。
a) Action中传数据:在Action类的请求处理方法中先获取各个作用域对象
ServletActionContext.getRequest();
ServletActionContext.getRequest().getSession();
ServletActionContext.getServletContext();
然后调用相应的setAttribute(String "键", Object 值);
b) 在JSP页面中取数据:可以使用EL表达式或代码片段来取出对应作用域中属性值。
c) 页面中的请求参数传递到Action中时,Action中直接定义对应名称的属性,并提供setter方法即可封装此数据。
2) 通过ActionContext实例来传递数据。 ActionContext针对每个正在执行Action的线程中绑定一份。
a) Action中通过ActionContext传递数据。
ActionContext提供了put(String "键", Object 值); //数据不会映射到HttpServletRequest中。
ActionContext提供的getSession().put(String "键", Object 值); //数据会自动映射到HttpSession中。
ActionContext提供的getApplication().put(String "键", Object 值); //数据会自动映射到ServletContext中。
b) 在JSP页面取数据:struts2推荐使用OGNL来取ActionContext中的数据。
1. Struts2中的OGNL的使用。
2. OGNL:对象图导航语言。通过OGNL表达式可以获取对象的属性,调用对象的方法,或构造出对象。
1) OGNL上下文中有一个根对象。这个根对象可以直接获取。不需要#。
2)支持常量:
字符串常量、字符常量、
数值常量:int、long、float、double
布尔常量:true、false
Null常量 : null
支持操作符:支持Java的所有操作符,还支持特有的操作符: ,、 {}、in、not in;
Struts2中的OGNL:
1) Struts2将ActionContext设置为OGNL上下文,并将值栈(ValueStack)作为OGNL的根对象放置到ActionContext中。
2) Struts2总是把当前Action实例放置在值栈的栈顶。所以,在OGNL中引用Action中的属性也可以省略“#”。
常用标签
1) <s:property value="OGNL"/>
2) <s:date name="OGNL" format=""/>
3) <s:if test="OGNL"></s:if><s:elseif test="OGNL"></s:elseif><s:else></s:else>
★4) <s:iterator value="OGNL" status="vs">...</s:iterator>
5) <s:debug/>
struts2中#、%和$这三个符号的使用方法【摘自max struts2教程】
一、"#"的用法
1、 访问OGNL上下文和Action上下文,#相当于ActionContext.getContext();下表有几个ActionContext中有用的属性:
parameters 包含当前HTTP请求参数的Map #parameters.id[0]作用相当于request.getParameter("id")
request 包含当前HttpServletRequest的属性(attribute)的Map #request.userName相当于request.getAttribute("userName")
session 包含当前HttpSession的属性(attribute)的Map #session.userName相当于session.getAttribute("userName")
application 包含当前应用的ServletContext的属性(attribute)的Map #application.userName相当于application.getAttribute("userName")
attr 用于按request > session > application顺序访问其属性(attribute) #attr.userName相当于按顺序在以上三个范围(scope)内读取userName属性,直到找到为止
2、用于过滤和投影(projecting)集合,如books.{?#this.price<100};
3、构造Map,如#{'foo1':'bar1', 'foo2':'bar2'}。
二、"%"的用法
“%”符号的用途是在标志的属性为字符串类型时,计算OGNL表达式的值。例如在Ognl.jsp中加入以下代码:
<h3>%的用途</h3>
<p><s:url value="#foobar['foo1']" /></p>
<p><s:url value="%{#foobar['foo1']}" /></p>
三、"$"的用法
1、用于在国际化资源文件中,引用OGNL表达式
2、在Struts 2配置文件中,引用OGNL表达式
例如:
<action name="AddPhoto" class="addPhoto">
<interceptor-ref name="fileUploadStack" />
<result type="redirect">ListPhotos.action? albumId=${albumId}</result>
</action>