ognl表达式
ognl表达式
action对象是存储在值栈中的,页面提交的数据会通过ognl给action中的属性及对象赋值,页面取值的时候,也是通过ognl从值栈中获取。
所以数据的传输是通过struts内置的ognl表达式及类型转换来完成,struts内置了类型转换器,来转换八种基本的数据类型。其他的数据类型需要自己手动写类型转换器来实现。
内置的转换器可以转换日期,但是只支持yyyy-MM-dd这种类型,如果需要其他的日期类型,需要自己开发类型转换器。
地址类型转换器:将字符串类型的地址转成地址对象
应用场景:假设在action中有个属性,地址类,private Address comAddress;
jsp页面为了简化,并没有给address类的每个字段一个输入框,而是只给一个输入框<s:textfield name="comAddress" label="comAddress"/>
让用户输入公司地址的时候,输入如下信息:福建-厦门-思明-xx街道,
如果是这样用户提交jsp页面的时候,ognl会把该输入地址(字符串)赋值给comAddress对象,这样就会出现类型不匹配的报错,为了解决这个问题
必须把用户输入的字符串地址,转换为Address的类型。
当然如果jsp页面是给四个输入框,则可以不用转换器。
/**
* 地址类型转换器
*/
public class AddressConverter extends StrutsTypeConverter {
/**
* 将字符串转换成地址对象
*/
public Object convertFromString(Map context, String[] values, Class toClass) {
if(values != null && values.length > 0){
String str = values[0];
String[] ss = str.split("-");
if(ss != null && ss.length > 2){
Address a = new Address();
a.setProvince(ss[0]);
a.setCity(ss[1]);
a.setStreet(ss[2]);
return a ;
}
}
return null;
}
/**
* 将地址对象转换成字符串信息
*/
public String convertToString(Map context, Object o) {
if(o instanceof Address){
return o.toString();
}
return null;
}
}
xwork-conversion.properties配置文件。要放在src根目录下
struts2.model.Address=struts2.ognl.AddressConverter
地址类:
/**
* 地址类
*/
public class Address {
private String province;
private String city;
private String street;
public String getProvince() {
return province;
}
public void setProvince(String province) {
this.province = province;
}
public String getCity() {
return city;
}
public void setCity(String city) {
this.city = city;
}
public String getStreet() {
return street;
}
public void setStreet(String street) {
this.street = street;
}
public String toString() {
return province + "-" + city + "-" + street;
}
}
ActionContext与ognl
ognl表达式可以解析任何对象,值栈是其中之一,而且是默认值。
ActionContext包含了所有范围的数据,共六个(valuestack,request,session,application,attr,parameters)。
parameters:是request中的请求参数封装到一个map集合中的数据。
request:是request范围内attribute属性数据,即通过request.setAttribute()设置的数据。
session:是session范围内attribute属性数据,即通过session.setAttribute()设置的数据。
application:是application范围内的attribute属性数据,即通过application.setAttribute()设置的数据。
attr:是从page,request,session,application四个域中查找数据,按照这个查找顺序找到的第一个数据。
ValueStack:即action中的数据,action中的数据都会放到值栈中。
ognl表达式可以从这六个范围内取数据,每个ognl表达式解析的时候都需要一个根对象(即需要指定一个域,默认的根对象是值栈)。即默认是从值栈中取数据。
选取session作为根对象,#session['user'] 表示:从session中取key值为user的对象。相对于session.getAttribute("user"); 默认情况下ValueStack是根对象。
#表示命名对象, 除了可以用ognl取值,还可以用EL表达式。
ActionContext:数据中心,Threadlocal
{
Map context
}
OgnlContext实现了Map接口。
当struts接收到请求后,马上创建一个ActionContext,一个ValueStack和一个Action对象,Action对象会立刻放到值栈上用于ognl访问。并且会放到栈顶。
顶层对象会覆盖底层对象,有多种方法可以把对象放到栈顶上,比如push标签。
例如:假设在值栈ValueStack中有对象map,user,xxAction等等对象,map对象在栈顶
那么当jsp页面有<s:property value="name" />,就会在值栈中从上至下的去搜索name属性的值,如果map中有key为name的值,则返回,
如果没有继续往下搜索user对象,如果user对象有name属性,则返回,如果没有,继续往下搜索,如果xxAction中有name属性,则返回。
值栈中可以存放各种类型的数据。
ognl从值栈中取值的图示:

数据存储:
ServletActionContext.getRequest().setAttribute("username", "requestTom");
ServletActionContext.getRequest().getSession().setAttribute("username", "sessionTom");
ServletActionContext.getServletContext().setAttribute("username", "applicationTom");
ServletActionContext.getContext().getValueStack().set("username", "vsTom");
request.username = <s:property value="#request.username" /><br>从request对象中取值,request不是默认的跟对象,需要加# session['username'] = <s:property value="#session.username" /><br> application.username = <s:property value="#application.username" /><br>
attr.username = <s:property value="#attr.username" /><br> parameters.username = <s:property value="#parameters.username" /><br> vs.username = <s:property value="username" /><br>值栈对象默认是根对象,不需要用#
强制ognl表达式解析
例如,label中的值,正常情况下不需要解析,内容会作为一个字符串显示出来,但是如果要想用ognl表达式解析的话,用解析出来的值显示,必须用%{}
<!-- 强制ognl表达式解析 -->
<s:textfield name="#session.username" label="%{#request.username}"/><br>
name="#session.username"是从session中取username的值,
label="%{#request.username}"从request中取username的值
<s:textfield name="username" label="%{#request.username}"/><br>
name="username" 是从值栈中取值
Data标签:该类标签用于从值栈中读写数据
有:property,set,push,bean,action
property标签:

如果value找不到值,就会用default的值代替,
escape:设置是否需要html转义。默认是转义的,即默认会把< 转成<
<s:property value="'<hr>tom'" escape="false"/>
set标签:
<s:set var="applicationmyname" scope="application" value="#session['username']" />这句话的意思是在application域中存入一个值,key为applicationmyname,值为session域中的username属性值 <s:property value="#application.applicationmyname" /> 从application域中获取上面存入的值。
s:set:如果没有指定范围scope的值,则存放ActioinContext自身的map中,<br> <s:set var="applicationmyname" value="#session['username']" /> <s:property value="applicationmyname" />
这里取值的时候可以加#,也可以不加#,为什么???
因为ActionContext内存储了其他六个的map集合,而此时没有指定scope也是存储在ActionContext自身的map中,因此是与这六个map同级的。
ognl搜索的顺序是从六个域的map中找,都找不到,就在自身的map中找。
set标签是创建值的标签的新的引用,push标签是放置新的属性值到值栈上,当你需要对一个对象进行大量操作时会有用,标签结束后会从栈顶删除该属性,而且该对象只放到值栈上,而set标签可以放到任意的范围中。
s:push,将对象放到栈顶,标签结束后从栈顶删除<br> <s:push value="#application.username">把application中的username放到栈顶
<s:debug />通过改变这句的位置可以看出是否在栈顶
<s:property/>如果这么写,表示把栈顶的对象输出,而且会调用对象的tostring()方法 </s:push>
<s:debug />放到这里,已经从栈顶删除了,username就不在栈顶了
在struts2中很少像java web那样生成的数据通过request.或者session等对象的setAttribuate()方法存到域中,然后再jsp中获取。
struts2获取数据是通过ognl来访问值栈获取数据,而action运行的时候会加载到值栈中,是根元素,所有在struts2中一般是声明集合list,或者map等属性,
在action中对该属性赋值,在jsp中通过ognl即可获取。
action中数据初始化:
public class OgnlAction extends ActionSupport {
private List userList ;public List getUserList() {
return userList;
}
public void setUserList(List userList) {
this.userList = userList;
}
private void popUserList(){
userList = new ArrayList<User>();
User u = null ;
for(int i = 0 ; i < 10 ; i ++){
u = new User();
u.setId(1 + i);
u.setName("tom" + i);
u.setAge(20 + i);
userList.add(u);
}
}
}
jsp中通过iterator迭代标签取值:
iteratator标签会把当前要迭代的对象放到栈顶。使用iterator迭代标签迭代的时候,可以不加#,因为此时对象已在栈顶。
s:iterator,迭代集合属性的<br>
<table border="1">
<tr>
<td>count</td><td>index</td><td>name</td><td>age</td><td>状态</td>
<td>编辑</td><td>查看</td>
</tr>
<s:iterator value="userList" var="u" status="st">
<tr class='<s:property value="#st.even?'even':'odd'" />'>
<td><s:property value="#st.count" /></td>
<td><s:property value="#st.index" /></td>
<td><s:property value="#u.name" /></td>
<td><s:property value="#u.age" /></td>
<td>
<s:if test="#u.age < 23">少年</s:if>
<s:elseif test="#u.age > 27">老年</s:elseif>
<s:else>中年</s:else>
</td>
<td>
<s:url action="UserAction_edit" namespace="/user" var="editUrl">
<s:param name="id" value="#u.id" />
<s:param name="name" value="#u.name" />
</s:url>
该url标签的值为:/user/UserAction_edit?id=11&name=tom 该值存在editUrl中,在ActionContext大map中
注意:不带var会直接输出,带var会存在var对应的变量里,该变量在ActionContext维护的大map中
<a href='<s:property value="#editUrl"/>'>编辑</a>获取该url值
</td>
<td><a href='<s:url action="UserAction_view" namespace="/user"><s:param name="id" value="#u.id" /></s:url>'>查看</a></td>
</tr>
</s:iterator>
</table>
<s:url/>
假设当前访问的页面为:http://localhost:8080/HelloWorldTest/testaction?name=1&age=12
<s:url/>输出的值为/HelloWorldTest/testaction
<s:url includeParams="all"/>输出的值为/HelloWorldTest/testaction?name=1&age=12
<s:url includeParams="all" var="myurl"/> 一旦加上var,该标签就不在输出值了,而是把值存在ActionContext大map中,key为myurl <s:property value="myurl"/>输出myurl的值
s:i18n,s:text通常结合在一起使用,访问指定资源文件中的字符串<br>
<s:i18n name="struts2.action.RegAction_zh">
<s:text name="label.age" var="mytext"/>
</s:i18n>
<s:property value="#mytext"/>
ogn表达式l操作List,Array:
list[0],array[0]获取第一个元素
list[0].name获取第一个元素的name属性值
array.length
array.size
array.isEmpty
访问集合的大小以及是否为空
<s:property value="userList.size"/>
<s:property value="userList.isEmpty"/> 返回true或false
ognl表示定义集合<br>
<s:iterator value="{'tom1','hjerry3','tomca5','kking7','jerrli9'}"> 构建集合的时候,字符串要用单引号,因为外层有双引号
<s:property /><br> 遍历value中的各值
</s:iterator>
<br>------------------------------------------------------------------------<br>
ognl定义map集合<br> 创建map的时候前面要加#,创建list不用
<s:iterator value="#{100:'tom',200:'jerry',300:'kingkong'}">
key = <s:property value="key" /> : value = <s:property value="value" /><br>
</s:iterator>
ogn表达式操作map:
map['foo'] 或者map.foo
map['user'].name
map.size
array.isEmpty
集合的过滤
<s:iterator value="userList.{?#this.age > 25}"> 获取userList集合中年龄大于25的对象
<s:property/><br> 直接这么写,会调用对象的tostring方法,所以如果是对象,一定要重写tostring方法
</s:iterator>
集合的投影,即子集
<s:iterator value="userList.{name}">把userlist集合中的所有name提取出来,构成新的集合
<s:property/><br>会输出userlist中的各对象的name值
</s:iterator>
s:checkbox标签
s:checkbox<br> <s:checkbox name="married" label="婚否" /> 在action获取到该值为true或false,因此action中对应该字段要为boolean类型。
public class UiAction extends ActionSupport {
private static final long serialVersionUID = -6933309304624396640L;
/* 婚否 */
private boolean married = true;//如果没有checkbox拦截器,该值默认为true会出问题,有了checkbox拦截器就解决了这个问题public boolean isMarried() {
return married;
}
public void setMarried(boolean married) {
this.married = married;
}
}
struts提供了checkbox拦截器,其作用是什么?
防止checkbox对应的属性设置默认的值为true,会带来问题,因为当checkbox没有选中的时候,checkbox标签产生的input表单中就没有value=false的属性,提交到action也就没有该值,这样由于action中设置了默认的值为true,因此就出现了错误。
checkbox拦截器就解决了这个问题,它在表单中加入了一个隐藏域,无论选中与否都会在隐藏域中保存value值,解决了这个问题。
所以使用checkbox标签的时候,一定要引入checkbox拦截器。
s:select标签
1. 用集合绑定:
<s:select list="{'tom','jerry','kingkong'}" label="username" /> list是绑定的值,可以从后台获取,也可以在此写死
生成的html标签的option下拉框的text与value值均为list中的内容
如:
<select name="" id="">
<option value="tom">tom</option>
<option value="jerry">jerry</option>
<option value="kingkong">kingkong</option>
</select>
2. 用map绑定:
<s:select list="#{101:'tom',202:'jerry',303:'kingkong'}" label="username"
listKey="key" listValue="value"/> 绑定map集合,listkey指定显示的值,listvalue指定绑定的值
3. 通过后台 Action的userlist集合绑定:
<s:select list="userList" listKey="id" listValue="name" label="UserName"/>
??????如何设定select绑定的初始值?????:通过在action中增加属性selectId,同时在select标签中增加name="selectId"的属性,也后台绑定。
<s:select name="selectedId" list="userList" listKey="id" listValue="name" label="UserName"/>
public class UiAction extends ActionSupport {
private static final long serialVersionUID = -6933309304624396640L;
/* 婚否 */
private boolean married = true;
/* javabean集合 */
private List userList;
/* 选中的id */
private Integer selectedId = 10; 绑定select的默认选中值。
public List getUserList() {
return userList;
}
public void setUserList(List userList) {
this.userList = userList;
}
public String reg() {
popUserList();
return "success";
}
public String toRegView() {
System.out.println("toRegView");
return "loginView";
}
/**
* 保存数据,
*/
public String saveData(){
popUserList();
popProvinces();
return "showView" ;
}
/**
* 组装用户集合
*/
private void popUserList(){
userList = new ArrayList<User>();
User u = null ;
for(int i = 0 ; i < 10 ; i ++){
u = new User();
u.setId(1 + i);
u.setName("tom" + i);
u.setAge(20 + i);
userList.add(u);
}
}
public Integer getSelectedId() {
return selectedId;
}
public void setSelectedId(Integer selectedId) {
this.selectedId = selectedId;
}
}
s:radio标签:
<s:radio name="sex" list="#{0:'男',1:'女'}" label="性别" />
s:checkboxlist标签:
<s:checkboxlist name="hobby" list="#{0:'足球',1:'篮球',2:'乒乓球'}" label="爱好"/>
s:doubleselect标签:双选框联动标签
<s:doubleselect name="selectProvinceId" list="provinces" listKey="id" listValue="areaName"
doubleName="selectedCityId" doubleList="cities" doubleListKey="id" doubleListValue="areaName"
label="区域"/>
public class UiAction extends ActionSupport {
private static final long serialVersionUID = -6933309304624396640L;/*省集合*/
private List<Area> provinces ;
/**
* 保存数据,
*/
public String saveData(){
popUserList();
popProvinces();
return "showView" ;
}
/**
* 组装province集合数组
*/
private void popProvinces(){
provinces = new ArrayList<Area>();
Area p1 = new Area(1,"1号省");
p1.getCities().add(new Area(11,"1.1城市"));
p1.getCities().add(new Area(12,"1.2城市"));
p1.getCities().add(new Area(13,"1.3城市"));
Area p2 = new Area(2,"2号省");
p2.getCities().add(new Area(21,"2.1城市"));
p2.getCities().add(new Area(22,"2.2城市"));
p2.getCities().add(new Area(23,"2.3城市"));
provinces.add(p1);
provinces.add(p2);
}public List<Area> getProvinces() {
return provinces;
}
public void setProvinces(List<Area> provinces) {
this.provinces = provinces;
}
}
/**
* 区域类
*/
public class Area {
private Integer id ;
private String areaName ;
private String areaNo ;
private int type ;
/* 省所属城市集合 */
private List<Area> cities = new ArrayList<Area>();
public Area(Integer id, String areaName) {
this.id = id;
this.areaName = areaName;
}
public Area(Integer id, String areaName, String areaNo) {
this.id = id;
this.areaName = areaName;
this.areaNo = areaNo;
}
public Area(Integer id, String areaName, String areaNo, int type) {
this.id = id;
this.areaName = areaName;
this.areaNo = areaNo;
this.type = type;
}
public Integer getId() {
return id;
}
public void setId(Integer id) {
this.id = id;
}
public String getAreaName() {
return areaName;
}
public void setAreaName(String areaName) {
this.areaName = areaName;
}
public String getAreaNo() {
return areaNo;
}
public void setAreaNo(String areaNo) {
this.areaNo = areaNo;
}
public int getType() {
return type;
}
public void setType(int type) {
this.type = type;
}
public List<Area> getCities() {
return cities;
}
public void setCities(List<Area> cities) {
this.cities = cities;
}
}


浙公网安备 33010602011771号