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; } }