在Servlet中,我们都是把数据放在域对象中,然后在jsp页面中进行读取。那么在Struts2中,提供了另外一种存储机制,就是ֵջ,在action中把值放入值栈,在页面中进行读取。
action:每次访问的时候都会创建一次。
servlet:默认在第一次访问的时候创建,在生命周期里只会创建一次。
我们创建一个LifeAction来看看
@SuppressWarnings("serial") public class LifeAction extends ActionSupport { public LifeAction() { System.out.println("action创建了。。。。"); } @Override public String execute() throws Exception { // 没有操作 return NONE; } }
在该类中有一个构造方法,每次创建action的时候都会打印信息。
在struts.xml中配置action
<action name="life" class="com.codeliu.action.LifeAction"></action>
访问后,每次刷新页面,都会创建一个action对象。
每次创建一个action对象,都会跟随着一个值栈对象,也就是说每个action里面都有一个值栈。
// 获取ActionContext实例对象 ActionContext context = ActionContext.getContext(); // 获取值栈对象 ValueStack stack1 = context.getValueStack(); ValueStack stack2 = context.getValueStack(); // true 说明每个action只有一个值栈对象 System.out.println(stack1 == stack2);
上述代码通过ActionContext类获取一个ValueStack(值栈),同时说明了一个action里面只有一个值栈。
值栈分为两部分。第一部分是root,第二部分是context,通过给成程序加断点,我们可以看到如下的结果
context是OgnlContext的实例对象,root是CompoundRoot的实例对象,我们看看这两个类的定义
public class CompoundRoot extends ArrayList {......}
public class OgnlContext extends Object implements Map {......}
这说明root其实本质是一个ArrayList,而context本质是一个Map。
root中一般存放数值,而context一般存放对象的引用。
下面是context的存储示意图
通过debug标签,可以查看值栈的内部结构。(要使用Struts2的标签,必须在jsp开头引入标签库)
创建一个ValueStackAction.java
@SuppressWarnings("serial") public class ValueStackAction extends ActionSupport { @Override public String execute() throws Exception { return SUCCESS; } }
在struts.xml中配置Action
<action name="valueStack" class="com.codeliu.action.ValueStackAction"> <result name="success">/valueStack.jsp</result> </action>
然后创建一个valueStack.jsp
<body> <!-- 通过这个标签获取值栈的信息 --> <s:debug></s:debug> </body>
运行这个action之后就能看到一个链接,点进去就能看到值栈的状态
可以看到root内部的栈顶是一个action实例。
action对象里有一个值栈对象,值栈对象里有action引用。
向值栈中放数据有三种方法,分别是使用set方法,push方法和使用action的变量,下面来看一下。
(1)使用set方法
public class ValueStackAction extends ActionSupport { @Override public String execute() throws Exception { /** * 往值栈中放数据的三种方法 */ // 第一种 使用set方法 ActionContext context = ActionContext.getContext(); ValueStack stack = context.getValueStack(); stack.set("url", "www.codeliu.com"); return SUCCESS; } }
同样使用debug标签查看值栈的状态,发现栈顶元素变成了我们插入的数据Map
(2)使用push方法
@SuppressWarnings("serial") public class ValueStackAction extends ActionSupport { @Override public String execute() throws Exception { /** * 往值栈中放数据的三种方法 */ ActionContext context = ActionContext.getContext(); ValueStack stack = context.getValueStack(); // 第二种 使用push方法 stack.push("codeliu"); return SUCCESS; } }
再次查看,发现栈顶元素变成了一个String
(3)使用Action类的变量来放数据
@SuppressWarnings("serial") public class ValueStackAction extends ActionSupport { // 1. 定义变量 private String url; // 2. 记得要get方法 public String getUrl() { return url; } @Override public String execute() throws Exception { /** * 往值栈中放数据的三种方法 */ // 3.设置值 url = "www.codeliu.com"; return SUCCESS; } }
刷新查看值栈,结果如下
我们发现栈顶元素没有变,但action实例中多了一行。
一般我们使用第三种方法更多,因为这样更节省空间,同时使用第三种方法,记得提供get方法,不然无法插入。
现在我们看看使用OGNL+Struts2的标签从值栈中读取使用上面第三种方法放入的数据。
(1)读字符串的值
@SuppressWarnings("serial") /** * 从值栈中获取字符串、对象和List集合 * @author liu */ public class GetValueFromValueStackAction extends ActionSupport { private String url; public String getUrl() { return url; } public void setUrl(String url) { this.url = url; } public GetValueFromValueStackAction() {} @Override public String execute() throws Exception { url = "www.codeliu.com"; return SUCCESS; } }
在struts.xml中增加一条记录
<action name="getValue" class="com.codeliu.action.GetValueFromValueStackAction"> <result>/getvalue.jsp</result> </action>
创建getvalue.jsp
<body> <!-- OGNL集合Struts2的标签获取值栈中的值 --> <s:property value="url"/> </body>
启动tomcat后,进入网页就能看到输出了url的值。
(2)读对象的值
还记得我们上篇文章中有一个User类,里面有username和password属性,现在我们修改GetValueFromValueStackAction类
@SuppressWarnings("serial") /** * 从值栈中获取字符串、对象和List集合 * @author liu */ public class GetValueFromValueStackAction extends ActionSupport { private User user = new User(); public User getUser() { return user; } public void setUser(User user) { this.user = user; } public GetValueFromValueStackAction() {} @Override public String execute() throws Exception { user.setUsername("CodeTiger"); user.setPassword("123456"); return SUCCESS; } }
在getvalue.jsp中添加如下代码
<!-- 通过ognl获取user对象的值 --> <s:property value="user.username"/> <s:property value="user.password"/>
运行后,输出我们设置的值。
(3)读List集合的值
修改GetValueFromValueStackAction类的代码如下
@SuppressWarnings("serial") /** * 从值栈中获取字符串、对象和List集合 * @author liu */ public class GetValueFromValueStackAction extends ActionSupport { private User user = new User(); private List<User> list = new ArrayList<User>(); public User getUser() { return user; } public void setUser(User user) { this.user = user; } public List<User> getList() { return list; } public void setList(List<User> list) { this.list = list; } public GetValueFromValueStackAction() {} @Override public String execute() throws Exception { user.setUsername("CodeTiger"); user.setPassword("123456"); list.add(user); return SUCCESS; } }
读取值栈中的List,一般有三种方法
<!-- 获取List集合中的值 --> <!-- 第一种方式 --> <s:property value="list[0].username"/> <s:property value="list[0].password"/><br> <!-- 第二种方式 --> <s:iterator value="list"> <s:property value="username"/> <s:property value="password"/> </s:iterator><br> <!-- 第三种方式 --> <s:iterator value="list" var="user"> <!-- 每次遍历出来的user对象,都会放到context中去,使用OGNL取root中的数据, 可以不用#,但取context中的数据,则要加上# --> <s:property value="#user.username"/> <s:property value="#user.password"/> </s:iterator>
注意第三种方式
下篇文章讲OGNL,所以讲OGNL之前得先熟悉了解一下值栈,才能更好的使用OGNL。