在Struts2中,一个请求在终于到达Action的方法之前,Action对象本身会被压入ValueStack(实际上就是放到ValueStack的CompoundRoot中),所以Action对象是CompoundRoot中的一个元素。看以下的代码:
public class UserAction {
private String username;
private Integer age;
private boolean valid;
//查看用户的具体信息
public String detail(){
username = "张三";
age = 18;
valid = true;
return "detail";
}
在Action中,给Action的username/age/valid赋值。Detail页面例如以下:
username:<s:property value="username"/> <br/>
valid:<s:property value="valid"/> <br/>
age:<s:property value="age"/> <br/>
上述JSP页面将能正确将它们的值取出。<s:property value=”ognl表达式”/>。在s:property标签中的OGNL表达式,终于会交给ValueStack来解释。username就是一个OGNL表达式,意思是调用root对象的getUsername()方法。Struts2将自己主动搜索CompoundRoot中有哪些元素(从第0个元素開始搜索),检測这些元素是否有getUsername()方法,假设第0个元素没有getUsername()方法,将继续搜索第1、2、3……个元素是否有getUsername()方法。
在上面的样例中,CompoundRoot中仅仅有一个对象,就是userAction对象,而这个对象中正好有getUsername()方法,所以,上述JSP代码将可以将值正确取出。
再看以下的样例:
public class UserAction {
private String username;
private String name;
//查看用户的具体信息
public String detail(){
username = "张三";
name = "王五";
User u = new User();
u.setUsername("赵毅");
ActionContext.getContext().getValueStack().push(u);
return "detail";
}
在上面这个UserAction的代码中,我们直接调用ActionContext.getContext().getValueStack().push()方法,把一个User对象(这个对象拥有getUsername()和setUsername()方法)直接压入到ValueStack中,这时候,在ValueStack的CompoundRoot中将有两个元素:第0个元素是刚刚压入的user对象[赵毅],而第1个元素是userAction对象[张三],假设在JSP中使用以下的表达式来取值:
<s:property value=”username”/> ,那么输出的值将是“赵毅”!道理上面已经讲过了,struts2将会从第0个元素開始搜索CompoundRoot中的对象,第0个元素正是刚刚压入的那个user对象!
假设在JSP中使用<s:property value=”name”/>来取值,将取出“王五”,由于第0个元素user对象没有name属性,所以,会继续搜索第1个元素userAction对象,在这个对象中就有name属性了!
再看以下的代码:
public class UserAction {
private String username;
//查看用户的具体信息
public String detail(){
username = "张三";
List list = new ArrayList();
for(int i=0; i<10; i++){
User user = new User();
user.setUsername("User"+i);
list.add(user);
}
ActionContext.getContext().put("users", list);
User u = new User();
u.setUsername("赵毅");
ActionContext.getContext().getValueStack().push(u);
return "detail";
}
相应的JSP例如以下:
1: <s:property value="username"/> <br/>
2: <s:iterator value="#users">
3: <s:property value="username"/>
4: <s:property value="#root[2].username"/><br/>
5: </s:iterator>
6: <s:property value="username"/>
7: <s:property value="#root[1].username"/> <!-- 张三 -->
依据刚才的演示样例,我们知道,第1行的username是“赵毅”(由于JSP在运行这行代码的时候,CompoundRoot中有两个元素:第0个是“user对象赵毅”,第1个是“userAction对象张三”),因此第1行的username将取出CompoundRoot中第0个元素的username属性:赵毅
第2行代码是iterator标签,仅仅定义了一个value属性,iterator标签将循环訪问users这个List中的User对象,并把当前循环的user对象压入到CompoundRoot中!所以,在第3行和第4行代码被运行的时候,CompoundRoot中总共同拥有3个元素:第0个元素是被iterator标签压入的当前循环的user对象;第1个元素是“user对象赵毅”;第2个元素是“userAction对象张三”,因此第3行代码的运行结果就是输出“UserX”,即当前循环的user对象的username属性!iterator标签将会依次取出List中的user对象,并不断压入/弹出user对象(每次循环,都将运行一遍压入/弹出)。而第4行代码取第2个元素的username属性,即userAction对象的username属性:张三。
第5行代码运行完毕之后,在CompoundRoot中将剩下2个元素,与第2行代码被运行之前一样。所以,第6行代码的输出和第1行代码的输出结果是一样的,而第7行代码将取出userAction对象的username属性:张三