在上篇(Hello World篇)中,我们呈现了一个简单的欢迎界面。在本篇中,我们将创建链接到应用其他Action的链接。
Web应用程序与传统的Web站点有所不同,在Web应用程序中我们可以创建动态的返回。为了更容易的在页面上获得动态数据,本框架提供了一些标签集。其中有些标签模拟了标准的HTML标签,但是提供了一些额外功能。还有一些框架提供了非标准的、但非常有用的控制功能。
Struts标签的一个使用是创建一个到另外的Web资源的访问链接,特别是到本地的其他资源的链接。
说明:虽然HTML为创建超链接提供了简单的标签,HTML标签常常需要我们包括一些冗余的信息。并且HTML标签不能很容易的得到本框架的动态数据。
一. 链接标签
在Web应用程序中的一个常用的功能是链接到其他页面,现在让我们为前面章节的欢迎界面添加到其他Action的链接。
二. 代码
显示注册与登录的jsp的代码如下:
<%@ taglib prefix="s" uri="/struts-tags" %>
<html>
<head>
<title>Welcome</title>
<link href="<s:url value="/css/tutorial.css"/>" rel="stylesheet"
type="text/css"/>
</head>
<body>
<h3>Commands</h3>
<ul>
<li><a href="<s:url action="Register"/>">Register</a></li>
<li><a href="<s:url action="Logon"/>">Sign On</a></li>
</ul>
</body>
</html>
运行效果如下:
另一个常用的功能是使用链接来改变语言,在前章的HelloWorld页面中,让我们改变用户的语言,并根据对应的应用程序资源来呈现信息。代码如下:
<body>
<h2><s:property value="message"/></h2>
<h3>Languages</h3>
<ul>
<li>
<s:url id="url" action="HelloWorld">
<s:param name="request_locale">en</s:param>
</s:url>
<s:a href="%{url}">English</s:a>
</li>
<li>
<s:url id="url" action="HelloWorld">
<s:param name="request_locale">es</s:param>
</s:url>
<s:a href="%{url}">Espanol</s:a>
</li>
</ul>
</body>
运行效果如下:
一. 代码是如何工作的?
上述例子中的“%{url}”将会被s:url标签来求得对应的值。在Welcome和HelloWorld页中,我们使用了两种不同的链接标签。我们可以创建如下标签:
l 资源链接
l 直接的链接
l 带参数的链接
下面让我们分别看一下它们的使用:
1. 资源链接
首先在jsp的HEAD元素中,我们使用url标签来将资源引入到页面中,代码如下:
<link href="<s:url value="/css/tutorial.css"/>"
rel="stylesheet" type="text/css"/>
注意:引用时绝对的。我们可以移动该页面到其他路经而不用担心相对路径。
2.直接链接
我们可以使用链接标签来定向到Action,实例代码如下:
<li><a href="<s:url action="Register"/>">Register</a></li>
当链接运作的时候,链接标签能够自动追加正确的扩展,因此我们不需要在应用程序中嵌入。这个标签也将会用会话id来编码链接的URL,因此Java的会话在请求之间能够保持。
3.带有参数的链接
在上述的有关改变语言的HelloWorld页中,我们可以使用带有参数信息的url标签来创建带有参数的链接,该段代码如下:
<s:url id="url" action="Welcome">
<s:param name="request_locale">en</s:param>
</s:url>
<s:a href="%{url}">English</s:a>
param标签将会在Welcome的Action的url后增加信息:“?request_locale=en”。这个标签接着将“url”引用注入到超链接中。
说明:通过增加param标签能够增加任意数量的标签。
二. 通配符映射
在上述例子的Welcome页面中,除了链接之外尚未添加任何内容,我们不需要添加Action类。但是,我们仍然需要添加一个映射,以便我们可以使用action URI(如果我们向映射到action,而不是页面,我们在后面也可以很容易的添加Action类),映射代码如下:
<action name="Welcome" >
<result>/tutorial/Welcome.jsp</result>
</action>
当我们创建应用程序的时候,我们常常需要直接链接到页页面,为了使原型更加容易,我们可以将Welcome的进入改为通配符,修改后的映射如下:
<action name="*" >
<result>/tutorial/{1}.jsp</result>
</action>
这是代码是如何工作的呢?
如果找不到映射,本框架将会做如下工作:
l 将“Welcome”映射为星号
l 将“Welcome”代替result中对应的“{1}”
同样的,如果在映射文件中找不到“Login”的映射,将会映射到“/tutorial /Login.jsp”页面。
说明:通配符映射能够使得你能够创建你自己的规约,以便你能够避免冗余的配置。
三. 数据输入表单
大多数应用程序都使用一些数据进入表单。Struts标签使得输入表单更加容易,代码如下:
<%@ taglib prefix="s" uri="/struts-tags" %>
<html>
<head>
<title>Login</title>
</head>
<body>
<s:form action="Logon">
<s:textfield label="User Name" name="username"/>
<s:password label="Password" name="password" />
<s:submit/>
</s:form>
</body>
</html>
执行效果如下:
代码是如何工作的呢?
l JSP标签在页面顶部引入了Struts的标签库
l Struts标签:textfield、password和submit,每一个都写出了正确的Lable和控制类型
四. 需要记住的东西
编写web应用程序的最难的一部分是编写页面。本框架通过提供一系列的Struts标签使得编写页面更加容易。Struts标签能够获得框架提供的动态数据。标签减少了用来创建页面所需做的工作。
附录一
——标签使用指南
一. 通用标签
通用标签被用来在页面执行的时候,来控制执行流。这些标签同样允许数据不是从Action或者值栈中提取,例如本地化、JavaBeans、包括额外的URL或action执行。
l 控制类标签提供了控制流,例如if,else和iterator
l 数据类标签允许数据操作或创建,例如bean、push和i18n
控制类标签有if、elseIf、 else、append、generator、iterator、merge、sort、subset。
数据类标签有a、action、bean、date、debug、i18n、include、param、push、set、text、url、property。
分别介绍如下:
1. 控制类标签
1)if、elseif和else
描述:执行基本的控制流,if能单独使用,也可与else、elseif标签搭配使用。
参数:
名称 |
必选 |
默认值 |
求值 |
类型 |
描述 |
id |
否 |
|
是 |
String |
id用来引用元素。对于UI或者form标签,它的意义等同于HTML的id属性 |
test |
是 |
|
是 |
Boolean |
用来决定标签体师是否显示的表达式 |
使用举例:
<s:if test="%{false}">
<div>Will Not Be Executed</div>
</s:if>
<s:elseif test="%{true}">
<div>Will Be Executed</div>
</s:elseif>
<s:else>
<div>Will Not Be Executed</div>
</s:else>
2)append
该标签的工作是追加迭代器来,
例如,如果有三个迭代器(每个迭代器有三个元素)需要追加,下面展示了这个追加迭代器是如何排列的。
首先是进入第一个迭代器的第一个元素;
第二步是进入第一个迭代器的第二个元素;
第三步是进入第一个迭代器的第三个元素;
第四步是进入第二个迭代器的第一个元素;
第五步是进入第二个迭代器的第二个元素;
第六步是进入第二个迭代器的第三个元素;
第七步是进入第三个迭代器的第一个元素;
第八步是进入第三个迭代器的第二个元素;
第九步是进入第三个迭代器的第三个元素。
参数:
名称 |
必选 |
默认值 |
求值 |
类型 |
描述 |
id |
否 |
|
是 |
String |
如果提供了该id的值,将会具有追加迭代器存储到堆栈上下文中的合成结果 |
使用举例:
Action类的代码:
private List myList1;
private List myList2;
private List myList3;
public String execute() throws Exception {
myList1 = new ArrayList();
myList1.add("1");
myList1.add("2");
myList1.add("3");
myList2 = new ArrayList();
myList2.add("a");
myList2.add("b");
myList2.add("c");
myList3 = new ArrayList();
myList3.add("A");
myList3.add("B");
myList3.add("C");
return "done";
}
public List getMyList1() { return myList1; }
public List getMyList2() { return myList2; }
public List getMyList3() { return myList3; }
jsp页面的代码:
<s:append id="myAppendIterator">
<s:param value="%{myList1}" />
<s:param value="%{myList2}" />
<s:param value="%{myList3}" />
</s:append>
<s:iterator value="%{#myAppendIterator}">
<s:property />
</s:iterator>
3)generator
描述:创建一个基于提供的值的迭代器。
注意:产生的迭代器将常常被推入堆栈顶部,而在标签结束的时候被推出。
参数:
名称 |
必选 |
默认值 |
求值 |
类型 |
描述 |
converter |
否 |
|
是 |
org.apache.struts2.util.IteratorGenerator.Converter |
将从值中分析的字符串转换为一个对象的转换器 |
count |
否 |
|
是 |
Integer |
在迭代器中的最大值 |
id |
否 |
|
是 |
String |
如果提供了id,它将会用来存储产生的迭代器到页面上下文 |
separator |
是 |
|
是 |
String |
分隔符用来将迭代器中的值分开 |
val |
是 |
|
是 |
String |
用来解析成迭代器的源 |
举例:
例1:
<pre>
例一:
产生一个简单的迭代器
<s:generator val="%{'aaa,bbb,ccc,ddd,eee'}">
<s:iterator>
<s:property /><br/>
</s:iterator>
</s:generator>
</pre>
这里产生了一个迭代器,并且使用iterator标签将它打印出来
例二:
<pre>
产生一个带有count属性的迭代器
<s:generator val="%{'aaa,bbb,ccc,ddd,eee'}" count="3">
<s:iterator>
<s:property /><br/>
</s:iterator>
</s:generator>
</pre>
这里产生了一个迭代器,但是只有其中的三个元素是可用的,这三个分别是aaa、bbb和ccc
例三:
<pre>
产生一个带有id属性的迭代器
<s:generator val="%{'aaa,bbb,ccc,ddd,eee'}" count="4" separator="," id="myAtt" />
<%
Iterator i = (Iterator) pageContext.getAttribute("myAtt");
while(i.hasNext()) {
String s = (String) i.next(); %>
<%=s%> <br/>
<% }
%>
</pre>
产生了一个迭代器,并且将它存入页面上下文的指定的id(myAtt)属性中。
例四:
<pre>
带有converter属性的generator标签
<s:generator val="%{'aaa,bbb,ccc,ddd,eee'}" converter="%{myConverter}">
<s:iterator>
<s:property /><br/>
</s:iterator>
</s:generator>
public class GeneratorTagAction extends ActionSupport {
....
public Converter getMyConverter() {
return new Converter() {
public Object convert(String value) throws Exception {
return "converter-"+value;
}
};
}
...
}
</pre>
产生的这个迭代器,它的每个元素由提供的转换器获得。在这个转换器中,只是为每个元素增加了“converter-”。
this converter, it simply add "converter-" to each entries.
4)iterator
描述:迭代器将会迭代值。一个可迭代的值可以是java.util.Collection,也可以是java.util.Iterator。
参数:
名称 |
必选 |
默认值 |
求值 |
类型 |
描述 |
id |
否 |
|
是 |
String |
id用来引用元素。对于UI和表单标签,它与HTML的id标签相当 |
status |
否 |
否 |
是 |
Boolean |
如果该值被指定,一个迭代状态的实例将会在每一个迭代中被推入堆栈中 |
value |
否 |
|
是 |
String |
用来进行迭代的迭代源,否则对象本身将会被放入新的产生的列表中 |
举例:
下面的例子取回在值栈中的当前对象的getDays()所返回的值,<s:property/>标签打印出当前迭代器的值,代码如下:
<s:iterator value="days">
<p>day is: <s:property/></p>
</s:iterator>
在下面的例子中使用Bean标签并且将它存入ActionContext中。iterator标签将会从ActionContext中取回对象而后调用它的getDays()方法。状态属性常常用来创建IteratorStatus对象,在这个例子中,它的odd()方法用来改变行的颜色。
<s:bean name="org.apache.struts2.example.IteratorExample" id="it">
<s:param name="day" value="'foo'"/>
<s:param name="day" value="'bar'"/>
</s:bean>
<p/>
<table border="0" cellspacing="0" cellpadding="1">
<tr>
<th>Days of the week</th>
</tr>
<p/>
<s:iterator value="#it.days" status="rowstatus">
<tr>
<s:if test="#rowstatus.odd == true">
<td style="background: grey"><s:property/></td>
</s:if>
<s:else>
<td><s:property/></td>
</s:else>
</tr>
</s:iterator>
</table>
下个例子将进一步展示status属性的使用,使用通过OGNL从action类取得的DAO,成员的迭代以及它们的使用(在安全的上下文中),last()方法中指明了当当前的对象是迭代器的最后一个可用的对象,如果不是,我们需要使用逗号来分隔用户,代码如下:
<s:iterator value="groupDao.groups" status="groupStatus">
<tr class="<s:if test="#groupStatus.odd == true ">odd</s:if><s:else>even</s:else>">
<td><s:property value="name" /></td>
<td><s:property value="description" /></td>
<td>
<s:iterator value="users" status="userStatus">
<s:property value="fullName" /><s:if test="!#userStatus.last">,</s:if>
</s:iterator>
</td>
</tr>
</s:iterator>
下一个例子在一个action collection上迭代,并且将每一个迭代的值传给另一个action。这里的诀窍在于使用”[0]”。它获得当前的值并且将值传入edit action。使用”[0]”与使用<s:property/>具有相同的效果(但是,后者,在param标签内部不起作用)。代码如下所示:
<s:action name="entries" id="entries"/>
<s:iterator value="#entries.entries" >
<s:property value="name" />
<s:property />
<s:push value="...">
<s:action name="edit" id="edit" >
<s:param name="entry" value="[0]" />
</s:action>
</push>
</s:iterator>
下例使用iterator标签来模拟一个简单的循环,循环了5次,代码如下:
<s:iterator status="stat" value="{1,2,3,4,5}" >
<!—获得当前的index(从0开始) -->
<s:property value="#stat.index" />
<!— 获得当前堆栈的值 -->
<!—当前的迭代值(0, 1, ... 5) -->
<s:property value="top" />
</s:iterator>
5)merge
描述:它是MergeIterator标签的组件,它的工作是合并迭代器和对合并后的迭代器的后续调用,它将使得每一个合并的迭代器有机会展示它的元素,接着下一个调用将会允许下一个迭代器来展示它的元素。一旦最后一个迭代器已展示完它的所有元素,第一个迭代器又能够开始展示它的元素(除非元素已经用尽)。
从内部来说,任务将委托给MergeIteratorFilter去做。
下面展示了3个列表的合并,其中每个列表有3个元素,步骤如下:
1. 展示第一个列表的第一个元素;
2. 展示第二个列表的第一个元素;
3. 展示第三个列表的第一个元素;
4. 展示第一个列表的第二个元素;
5. 展示第二个列表的第二个元素;
6. 展示第三个列表的第二个元素;
7. 展示第一个列表的第三个元素;
8. 展示第二个列表的第三个元素;
9. 展示第三个列表的第三个元素;
参数:
名称 |
必选 |
默认值 |
求值 |
类型 |
描述 |
id |
否 |
|
是 |
String |
合并后的迭代器的值将会存储在堆栈上下文的id |
举例:
Action类代码:
public class MergeIteratorTagAction extends ActionSupport {
private List myList1;
private List myList2;
private List myList3;
public List getMyList1() {
return myList1;
}
public List getMyList2() {
return myList2;
}
public List getMyList3() {
return myList3;
}
public String execute() throws Exception {
myList1 = new ArrayList();
myList1.add("1");
myList1.add("2");
myList1.add("3");
myList2 = new ArrayList();
myList2.add("a");
myList2.add("b");
myList2.add("c");
myList3 = new ArrayList();
myList3.add("A");
myList3.add("B");
myList3.add("C");
return "done";
}
}
jsp页代码:
<s:merge id="myMergedIterator1">
<s:param value="%{myList1}" />
<s:param value="%{myList2}" />
<s:param value="%{myList3}" />
</s:merge>
<s:iterator value="%{#myMergedIterator1}">
<s:property />
</s:iterator>
二. UI标签
三. 主题与模板标签
四. 标签引用
五. Ajax标签
六. 标签语法
标签被设计用来显示动态的数据。为了创建输入域来显示属性“postalCode”,我们需要将“postalCode”传给textfield标签。
下面创建了一个动态的输入域:
<s:textfield name="postalCode"/>
如果在值栈中存在“postalCode”属性,它的值将会被放入该输入域中。当输入被提交到框架之后,它的值将会被放置到“postalCode”属性中。
有时候,我们想要传动态的数据到标签中。例如,我们可能需要用输入域来显示一个label,我们可能想要从应用程序的message资源中。相应地,框架将会解析在标签属性中的表达式。因此在运行时我们能够合并动态的数据到标签的属性中。表达式是像“%{…}”这样的。任何一个在其中嵌入的文本被作为表达式来计算。
使用一个表达式来设置label的例子:
<s:textfield key="postalCode.label" name="postalCode"/>
表达式语言(OGNL)使得我们能够调用方法和计算属性。getText()方法由ActionSupport(大多数Action的基类)提供。我们可以调用任何一个表达式提供的方法,包括getText方法。
非String型的属性
HTTP协议是基于文本的,但是有一些标签时非String类型的,例如bool和int。为了能够直接使用非String型的属性,本框架将所有的非String类型的属性作为一个表达式处理,在这种情况下,你不需要使用任何转义符(但是,如果你使用了转义符,框架也会将其跳过)。
计算boolean型的例子:
<s:select key="state.label" name="state" multiple="true"/>
一旦multiple属性跟一个boolean属性对应起来,框架不会将它作为一个String来解释,这个值将会被作为一个表达式来计算并且自动地将其转换为boolean值。
因为很容易忘记哪个属性是String,哪个属性是非String型的,你仍然可以使用转义符。
计算boolean值(带有转义符的):
<s:select key="state.label" name="state" multiple="%{true}"/>
带有属性的:
<s:select key="state.label" name="state" multiple="allowMultiple"/>
既带有转义符又带有属性的:
<s:select key="state.label" name="state" multiple="%{allowMultiple}"/>
值是一个对象
更通常的情况是,属性值是自动放入的,因为name属性通常告诉框架哪个属性来调用方法来set值。但是,如果需要直接设置值,建议那个值是Object型而不是String型。
注意:因为值是非String型的,无论传入什么,都是将他作为表达式来处理——而不是一个字面的String。
可能导致错误的例子:
<s:textfield key="state.label" name="state" value="CA"/>
如果textfield被传入 “CA”,本框架将会名字为getCa的属性。通常情况下,这不是我们的本意,我们想要做的是传送一个字符串。在表达式语言中,文字需要被置为单引号之间。
以正确的方式传入一个文字值:
<s:textfield key="state.label" name="state" value="%{'CA'}" />
另一种方法是使用value=”’CA’”,但是在这种情况下,推荐使用表达式符号。
标签属性使用如下三种规则计算:
1. 所有的String属性都被解析成“%{…}”符号;
2. 所有的非String属性没有被解析,而是直接按照表达式来算;
3. 规则2的异常情况是非String型的属性使用转义符号“{%{}”,符号被作为多余的符号被忽略,而只是计算内容。
表示式语言符号
1.在Freemarker、Velocity或者JSTL的表达式语言的JavaBean对象的标准文本
<p>Username: ${user.username}</p>
2.在值栈中的一个username属性
<s:textfield name="username"/>
3. 引用值栈中的属性的另一种方式
<s:url id="es" action="Hello">
<s:param name="request_locale">
es
</s:param>
</s:url>
<s:a href="%{es}">Espanol</s:a>
4. 在Session Context中获得user对象的userName属性
<s:property name="#session.user.username" />
5. 在一个静态map中,像("username","trillian")一样
<s:select label="FooBar" name="foo" list="#{'username':'trillian', 'username':'zaphod'}" />
下一篇:编写Action
上一篇:HelloWorld篇