c:forEach的標籤的語法定義
<c:forEach var="name" items="expression" varStatus="name"
begin="expression" end="expression" step="expression">
body content
</c:forEach>
<c:forEach>標籤具有以下一些屬性:
l var:迭代參數的名稱。在迭代體中可以使用的變量的名稱,用來表示每一個迭代變量。類型爲String。
l items:要進行迭代的集合。對於它所支持的類型將在下面進行講解。
l varStatus:迭代變量的名稱,用來表示迭代的狀態,可以訪問到迭代自身的信息。
l begin:如果指定了items,那麼迭代就從items[begin]開始進行迭代;如果沒有指定items,那麼就從begin開始迭代。它的類型爲整數。
l end:如果指定了items,那麼就在items[end]結束迭代;如果沒有指定items,那麼就在end結束迭代。它的類型也爲整數。
l step:迭代的步長。
<c:forEach>標籤的items屬性支持Java平臺所提供的所有標準集合類型。此外,您可以使用該操作來迭代數組(包括基本類型數組)中的元素。它所支持的集合類型以及迭代的元素如下所示:
l java.util.Collection:調用iterator()來獲得的元素。
l java.util.Map:通過java.util.Map.Entry所獲得的實例。
l java.util.Iterator:迭代器元素。
l java.util.Enumeration:枚舉元素。
l Object實例數組:數組元素。
l 基本類型值數組:經過包裝的數組元素。
l 用逗號定界的String:分割後的子字符串。
l javax.servlet.jsp.jstl.sql.Result:SQL查詢所獲得的行。
不論是對整數還是對集合進行迭代, <c:forEach>的varStatus屬性所起的作用相同。和var屬性一樣,varStatus用於創建限定了作用域的變量(改變量只在當前標籤體內起作用)。不過,由varStatus屬性命名的變量並不存儲當前索引值或當前元素,而是賦予javax.servlet.jsp.jstl.core.LoopTagStatus類的實例。該類包含了一系列的特性,它們描述了迭代的當前狀態,如下這些屬性的含義如下所示:
l current:當前這次迭代的(集合中的)項。
l index:當前這次迭代從0開始的迭代索引。
l count:當前這次迭代從1開始的迭代計數。
l first:用來表明當前這輪迭代是否爲第一次迭代,該屬性爲boolean類型。
l last:用來表明當前這輪迭代是否爲最後一次迭代,該屬性爲boolean類型。
l begin:begin屬性的值。
l end:end屬性的值
l step:step屬性的值
<c:forEach /c>和ResultSet結合
微軟的.NET平臺上面的數據訪問有一個特點,就是數據查詢的結果,可以放在內存中,以XML格式進行描述,不需要一直與數據庫保持在線連接,用DataSet + Data Adapter來實現!
而在JDBC中,我們通常使用javax.sql.ResultSet類來存放放回的數據,它的流程和生命週期如下:
使用ResultSet來返回數據庫查詢結果 | |||||||
Client | --> | Connection | --> | Statement | --> | JDBC Driver | --+ |
Database | |||||||
Client | <-- | Parsing | <-- | ResultSet | <-- | JDBC Driver | --+ |
Connection lifecycle | |||||||
ResultSet lifecycle | |||||||
我們可以看到,這樣會長期佔用數據庫連接的資源,是一個有點不爽的問題...
其實,在JSTL中提供了另外一種機制,讓我們在返回查詢結果到表示層的時候,可以做到離線使用!它就是javax.servlet.jsp.jstl.sql.Result類!
示範代碼如下:
<%@ taglib prefix="c" uri="/WEB-INF/c.tld" %>
<%@ page language="java" import="java.util.*" pageEncoding="gb2312"%>
<%@page import="java.sql.Connection"%>
<%@page import="java.sql.DriverManager"%>
<%@page import="java.sql.ResultSet"%>
<%@page import="java.sql.SQLException"%>
<%@page import="java.sql.Statement"%>
<%@page import="javax.servlet.jsp.jstl.sql.Result"%>
<%@page import="javax.servlet.jsp.jstl.sql.ResultSupport"%>
<%
// 暫且把這個下面的內容看作多層架構中的DAO好了,我偷懶了!
Connection conn = null;
Statement stmt = null;
ResultSet rs = null;
try {
//開始與數據庫作查詢
Class.forName("com.microsoft.jdbc.sqlserver.SQLServerDriver").newInstance();
String url="jdbc:microsoft:sqlserver://localhost:1433;DatabaseName=db_news";
//pubs爲你的數據庫的
String user="sa";
String password="";
conn= DriverManager.getConnection(url,user,password);
stmt=conn.createStatement(ResultSet.TYPE_SCROLL_SENSITIVE,ResultSet.CONCUR_UPDATABLE);
String sql="select * from tb_news";
rs = stmt.executeQuery(sql);
//把ResultSet轉化成Result
Result userData = ResultSupport.toResult(rs);
//當我們把結果放到某個Model bean後,就可以關閉數據庫連接了
//爲了簡化,我們暫且把pageContext看作Model Bean好了
pageContext.setAttribute("userData", userData);
}
catch (Exception ex) {
// handle any errors
System.out.println("SQLException: " + ex.getMessage());
}
finally {
try {
if (rs != null) {
rs.close();
}
if (stmt != null) {
stmt.close();
}
if (conn != null) {
conn.close();
}
}
catch (SQLException ex) {
System.out.println("SQL Exception: " + ex.getMessage());
}
}
//DAO的邏輯結束51
%>
<html>
<head>
<title>test</title>
</head>
<body>
<TABLE border=1>
<TR>
<TD>文章ID</TD>
<TD>文章名</TD>
<TD>文章內容</TD>
<TD>文章類型</TD>
<TD>文章類型</TD>
<TD>最後修改時間</TD>
</TR>
<c:forEach items="${userData.rows}" var="user">
<TR>
<TD><c:out value='${user.ID}'/></TD>
<TD>${user.Title}</TD>
<TD>${user.Content}</TD>
<TD>${user.Type}</TD>
<TD>${user.Style}</TD>
<TD>${user.IssDate}</TD>
</TR>
</c:forEach>
<TABLE>
</body>
</html>
結果:
文章ID | 文章名 | 文章內容 | 文章類型 | 文章類型 | 最後修改時間 |
81 | 老虎 | 老虎和狼打仗了。 | 野生動物 | 大自然動物開始稀少 | 2006-02-07 08:40:00.0 |
82 | 周迅要來長 | 明天下午周迅來長春簽名售書。 | 娛樂 | 影星 | 2006-02-07 |