Ognl表達式基本原理和使用方法
1.Ognl表達式語言
1.1.概述
OGNL表達式
OGNL是Object Graphic Navigation Language(對象圖導航語言)的縮寫,他是一個開源項目。Struts框架使用OGNL作為默認的表達式語言。
OGNL優勢
- 支持對象方法調用,如:×××.doSomeSpecial();
-
支持類靜態的方法調用和值訪問,表達式的格式
@[類全名(包括包路徑)]@[方法名 | 值名],例如:
@java.lang.String@format('foo %s', 'bar')
或@tutorial.MyConstant@APP_NAME; -
支持賦值操作和表達式串聯,
如price=100, discount=0.8,calculatePrice(),這個表達式會返回80;
- 訪問OGNL上下文(OGNL context)和ActionContext;
-
操作(創建)集合對象。
總結:OGNL 有一個上下文(Context)概念,說白了上下文就是一個MAP結構,它實現了java.utils.Map 的接口。
Struts框架默認就支持Ognl表達式語言。(從struts項目必須引入ognl.jar包可以看出)
Ognl表達式語言的作用:
- jsp頁面取值用
- EL表達式語言,也用於頁面取值,是jsp頁面取值的標准(默認就可以使用)
- Ognl表達式語言,Struts標簽默認支持的表達式語言,必須配置Struts標簽用,不能離開Struts標簽直接使用,就是說Ognl必須在Struts中使用
- 對比來看,EL使用范圍更廣,項目中不限制使用哪一種,哪一種熟悉就使用哪一種
1.2.OgnlContext對象(了解)
OgnlContext對象是ognl表達式語言的核心。
但是項目中不會要求寫OgnlContext的代碼,Ognl標簽其實是調用了OgnlContext對象。所以只做了解即可。
OgnlContext對象在源碼中實現了Map接口:public class OgnlContext implements Map {……}
Ognl表達式語言取值,也是用java代碼取值的,原理就是使用OgnlContext和Ognl這兩個類,只需要記住,Ognl取根元素不用#號,取非根元素要使用#號:
OgnlContext類
硬編碼方式,了解OgnlContext對象,因為OgnlContext對象實現是Map接口,所有OgnlContext本質就是一個Map,可以使用map方法:
OgnlContext context = new OgnlContext();
context.put("uesr",user);
context.put("address",address);
context.setRoot(address);
Ognl類
Ognl類也是Ognl底層運行的代碼,常用的api如下:
Object obj1 = Ognl.parseExpression(“country”); 解析ognl表達式
Ognl.getValue(obj1, context, context.getRoot()); 獲取ognl的表達式值,obj1是上面一個api,其他兩個分別是創建的上下文對象以及一個不用修改的參數
Object obj2 = Ognl.parseExpression(“language.toUpperCase()”); 方法調用
Object obj3 = Ognl.parseExpression("@java.lang.Integer@toBinaryString(10)");等同於上面
Object obj4 = Ognl.parseExpression(“@@min(10,4)”); Math類的方法直接調用,靜態方法的調用
代碼示例如下:
package o_ognl; import ognl.Ognl; import ognl.OgnlContext; import ognl.OgnlException; import org.junit.Test; /** * OgnlContext用法 * 1.使用Ognl表達式語言取值,如果取非根元素的值,必須用#號 * 2.使用Ognl表達式語言取值,如果取根元素的值,不用#號 * 3.Ognl可以調用靜態方法 */ public class OgnlDemo { //非根元素 @Test public void testOgnl1() throws OgnlException { //創建一個Ognl上下文對象 OgnlContext context = new OgnlContext(); /** * 1.OgnlContext放入基本變量數據 */ //放入數據 context.put("cn","China"); //獲取數據(map) String value = (String)context.get("cn"); System.out.println(value); /** * 2.OgnlContext放入對象數據 */ //創建對象,設置對象屬性 User user = new User(); user.setId(100); user.setName("Jack"); //【往非根元素放入數據,取值的時候表達式要用“#”】 context.put("user",user); //獲取對象屬性 //使用這種方式也可以獲取 Object s = context.get("user"); System.out.println(s); //使用Ognl表達式來獲取 //舉例:例如標簽<s:a value="#user.id">取值,實際上就是運行了下面的代碼獲取的 //先構建一個Ognl表達式,再解析表達式 Object ognl = Ognl.parseExpression("#user.id");//構建Ognl表達式 Object value1 = Ognl.getValue(ognl, context, context.getRoot());//解析表達式 System.out.println(value1); User user1 = new User(); user1.setId(100); user1.setName("Jack"); context.setRoot(user1); Object ognl1 = Ognl.parseExpression("id");//構建Ognl表達式 Object value2 = Ognl.getValue(ognl1, context, context.getRoot());//解析表達式 System.out.println(value2); } //根元素, @Test public void testOgnl2() throws OgnlException { OgnlContext context = new OgnlContext(); User user1 = new User(); user1.setId(100); user1.setName("Jack"); context.setRoot(user1); //根元素直接使用id,不需要加#號 Object ognl1 = Ognl.parseExpression("id");//構建Ognl表達式 Object value2 = Ognl.getValue(ognl1, context, context.getRoot());//解析表達式 System.out.println(value2); } //ognl對靜態方法調用的支持 @Test public void testOgnl3() 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); } }
1.3.ValueStack對象
1.ValueStack即值棧對象
ValueStack實際是一個接口,在Struts2中利用Ognl時,實際上使用的是實現了該接口的OgnlValueStack類,這個類是Struts2利用Ognl的基礎
2.ValueStack特點
ValueStack貫穿整個Action的生命周期(每個Action類的對象實例都擁有一個ValueStack對象),即用戶每次訪問struts的action,都會創建一個Action對象、值棧對象、ActionContext對象,然后把Action對象放入值棧中;最后再把值棧對象放入request中,傳入jsp頁面。相當於一個數據的中轉站,在其中保存當前Action對象和其他相關對象。Struts2框架把ValueStack對象保存在名為“struts。valueStack”的request請求屬性中。
3.ValueStack存儲對象
代碼調試的時候,發現有一個root是compundRoot類,繼承ArrayList,保存的是action對象;還有一個OgnlContext是繼承Map,保存數據。
所以ValueStack存儲對象時是分兩個地方來存的,也即ValueStack對象的組成是由List棧和Map棧構成的:ObjectStack
:Struts把根元素,即action對象及全局屬性存入ObjectStack中---List
list棧主要存儲:action對象,Map對象(通過vs.set()設置),通過push方法設置的對象,以及其他代理對象
根元素的存儲示例:
//存儲值棧對象 ActionContext ac = ActionContext.getContext(); ValueStack vs = ac.getValueStack(); vs.set("user1",new User(100,"Jack1"));//Map vs.push(new User(100,"Jack2"));//棧頂
ContextMap
:Struts把各種各樣的映射關系(域數據)存入ContextMap中
Struts會把下面這些映射存入ContextMap中:
- parameter:該Map中包含當前請求的請求參數
- request:該Map中包含當前request對象中的所有屬性
- Session:該Map中包含當前Session對象中的所有屬性
- application:該Map中包含當前application對象中的所有屬性
-
attr:該Map按如下順序來檢索某個屬性:request,Session,application
非根元素Map中存放數據的方法示例://存儲值棧對象 ActionContext ac = ActionContext.getContext(); //映射數據 ac.getContextMap().put("request_data", "request_data"); ac.getSession().put("session_data", "session_data"); ac.getApplication().put("application_data","application_data");
從棧中取值的兩種方式:
//一、獲取值棧對象的兩種方式,是等價的 public void getVs() { //獲取值棧對象,方式1: HttpServletRequest request = ServletActionContext.getRequest(); ValueStack vs1 = (ValueStack) request.getAttribute("struts.valueStack"); /**************************************************/ //獲取值棧對象,方式2: ActionContext ac = ActionContext.getContext(); ValueStack vs2 = ac.getValueStack(); System.out.println(vs1==vs2);//true }
在jsp頁面中,對不同ValueStack中的不同類型取值方法不同,
如果是根元素取值,直接寫表達式;
非根元素(request,Session,application,att,parmeters)必須用#號,例#request.cn
1.4.JSP頁面中獲取ValueStack數據
<%@taglib prefix="s" uri="/struts-tags" %> <%@ page contentType="text/html;charset=UTF-8" language="java" %> <html> <head> <title>jsp頁面取值</title> </head> <body> index頁面 <%--頁面,必須要拿到ValueStack--%> <%--struts的調試標簽,可以觀測值棧數據--%> <s:debug/> <br/>1.取根元素的值<br/> <s:property value="user.id"/> <s:property value="user.name"/> <br/>2.取非根元素<br/> <s:property value="#request.cn"/> <s:property value="#request.request_data"/> <s:property value="#session.session_data"/> <s:property value="#application.application_data"/><br/> <%--attr按順序自動找request/session/application,找到后立刻返回--%> <s:property value="#attr.application_data"/> <%--獲取請求參數數據--%> <s:property value="#parameters.userName"/> </body> </html>
2.Struts標簽
struts標簽取值,就使用到ognl表達式語言。
2.1.Ognl使用方式
1.4.中使用的就是Ognl表達式取值。使用方式是:
1.引入<%@taglib prefix="s" uri="/struts-tags" %>
2.使用 <s:property value="user.name"/>
標簽獲取取值,取值的時候要注意根元素(全局變量)不用#號,其他的都用#號
還有一些標簽需要學習:
2.2.迭代標簽
Iterator:標簽用於對集合進行迭代,這里的集合包含List,Set和數組
- value:可選屬性,指定被迭代的集合,如果沒有設置該屬性,則使用ValueStack棧頂的集合。
- var:可選屬性,引用變量的名稱
-
status:可選屬性,該屬性指定迭代是的IteratorStatus實例,該實例包含一下一個方法;
int getCount(),返回當前迭代了幾個元素
int getIndex(),返回當前迭代元素的索引
boolean isEven(),返回當前迭代元素的索引是否是偶數
boolean isOdd(),返回當前迭代元素的索引是否是奇數
boolean isFirst(),返回當前迭代元素是否是第一個元素
boolean isLast(),返回當前迭代元素是否是最后一個元素
代碼示例:
<br/>1.list迭代<br/> <s:iterator var="user" value="#request.list" status="st"> <s:property value="#user.id"/> <s:property value="#user.name"/> <s:property value="#st.even"/><br/> </s:iterator> <br/>2.map迭代<br/> <s:iterator var="en" value="#request.map" status="st"> <s:property value="#en.key"/> <s:property value="#en.value.name"/> </s:iterator>
2.3.Ognl動態建立集合
<%--Ognl表達式可以取值,也可以構建動態集合--%> <br/>1.構建list集合<br/> <s:iterator var="str" value="{'a','b','c'}"> <s:property value="#str"/> </s:iterator> <br/>2.構建map集合<br/> <s:iterator var="en" value="#{'cn':'China','usa':'America'}"> <s:property value="en.key"/> <s:property value="en.value"/><br/> </s:iterator>
2.4.簡單UI標簽
<%@taglib prefix="s" uri="/struts-tags" %> <%--服務器標簽:最終被解析為html標簽--%> <s:form action="/user_login" method="POST" name="frmLogin"> <s:textfield name="user.userName" label="用戶名"/> <s:textfield name="user.pwd" label="密碼"/> <s:submit value="登錄"/> </s:form>
也可以給form指定主題,方法如下:
<!-- 服務器標簽 : 最終別解析為html標簽--> <s:form action="/user_login" method="post" name="frmLogin" id="frmLogin" theme="simple"> 用戶名:<s:textfield name="user.name"></s:textfield> 密碼:<s:textfield name="user.pwd"></s:textfield> <s:submit value="登陸"></s:submit> </s:form>
注意:
給form指定主題,form下所有的表單元素都應用此主題
對Struts標簽默認的主題樣式:default.xml/struts.ui.theme=xhtml
可以通過常量修改,改為簡單主題:
<!-- 修改主題 (當前項目所有的標簽都用此主題)--> <constant name="struts.ui.theme" value="simple"></constant>
2.5.Ognl表達式語言幾個符號
:獲取非根元素,動態創建map集合
$:配置文件取值
%:提供一個ognl表達式運行環境
<body>
<br/>獲取request域數據<br/>
<!-- property 標簽是對象類型的標簽,默認支持ognl表達式, 會從根元素去China名稱對應的值 --> <s:property value="China"/> <br/> <!-- 如果直接賦值,需要用單引號 --> <s:property value="'China'"/> <br/> <s:property value="%{#request.cn}"/> <br/> <!-- 值類型的標簽,value值默認就是值類型,不支持ognl表達式 --> 國家:<s:textfield name="txtCountry" value="%{#request.cn}"></s:textfield> </body>