基於 Struts 2 構建 WebSphere Portal 上的 Portlet 應用
Struts 2 是 MVC 框架發展的最新階段。Struts 2 從 WebWork 發展而來,而不是由 Struts1 演化而來,因此利用 Struts 2 開發和部署應用程序有很多不同於 Struts1 的地方,尤其是在開發和部署 Porlet 方面,Struts2 更是有著以往 Portlet 應用程序開發方式所無法比擬的優勢。本文的目的就是通過在 IBM 的 WebSpherePortal Server 上開發和部署一個基於 Struts 2 的 Porlet 應用,向讀者介紹利用 Struts2 進行 Portlet 應用開發的優勢和關鍵流程。
概述
WebSphere Portal Server5.1 及以上版本支持兩種 Portlet API:第一種是 IBMPortlet,這種 Portlet API 是 WebSphere PortalServer 專有的一種 Portlet API;第二種是符合 JSR168 標準的 PortletAPI。由於 JSR168 是一個開放的標準,因此符合 JSR168 標準的 Portlet 將更易於移植。
IBM 為 IBM Portlet API 和 JSR168API 分別實現了基於 Struts1 的 Portlet 開發框架,由於基於Struts1,這兩種 Portlet 框架 API 同 Portlet API 耦合緊密,尤其需要指出的是由於無論 IBM Portlet還是在 WebSphere Portal Server 上實現的 JSR168 標準的 Porlet API, 它們的介面都直接依賴於PortletRequest/PortletResponse 對象,這就使得程序移植和單元測試等變得比較困難。此外,我們在使用 Struts1開發 Servlet 應用時,習慣於將數據存放在 request 作用域中,通過頁面的跳轉將數據呈現到 jsp 視圖頁面。但是,這種做法在portlet 開發中是不可行的。與 servlet 的生命周期有所不同,portlet 存在操作響應階段和呈現階段。 在 portlet操作響應階段存放在 request 作用域的變數,在呈現階段就會失效。在原有 API 上解決這個問題既費時又不優雅,而 Struts 2 對Portlet 的支持將能夠很好的解決這些問題。
本文就是要通過一個簡單的示例應用程序的開發和部署過程,來展示 Struts 2 怎樣解決舊有的 PortletAPI 所無法克服的困難的。
本文的重點不在於開發一個 Struts2 Web 應用程序,而在於開發一個作為 Portlet 的Struts2 應用程序所需的的實現和配置。讀者可以了解到如何利用 Struts 2 來創建一個Portlet,這個 Portlet 將完全獨立於其所開發和部署的平台。
在示例應用程序的開發和部署中用到了下列產品:
Porlet 示常式序設計概述
示例應用程序是一個簡單的用戶登錄程序。合法的用戶將跳轉到的登錄成功頁面,登錄失敗的用戶則跳轉到登錄失敗頁面,並被要求輸入正確的用戶名和密碼。用戶可以自由的在 Portlet 的 View、Edit 和 Help 模式之間進行切換。應用程序視圖部分分為以下幾部分:
圖 1. 程序初始頁面
下圖是本文示例的 Action 與頁面的交互圖:
圖 2. Action 與頁面的交互圖利用 Struts2 實現 Portlet
在本文中,使用 Struts2 開發 Portlet 應用需要經歷以下步驟:
- 使用 RAD7 建立開發環境
- 生成 web.xml 配置文件
- 生成 portlet.xml 配置文件
- 編輯 jsp 文件
- 應用 Struts2
- 部署 Portlet 應用程序
- 訪問 Portlet 應用
使用 RAD7 建立開發環境
在 Rational Application developer 7中啟動一個 Portlet 項目,需要遵循下列步驟:
- 選擇新建一個 Portlet 項目,如圖 3 所示: 圖 3. 新建一個 Portlet 項目
- 輸入項目名 Struts2TestPortlet,目標運行時選擇 WebSphere Portal v5.1 或者更高版本,注意 Portlet API 選擇 JSR 168 Portlet, 注意勾掉創建 portlet 選項,點擊完成按鈕。如圖 4 所示: 圖 4. 設置項目屬性
生成 portlet 項目及其結構如圖 5 所示: 圖 5.portlet 項目及其結構
-
接下來到 Apache 官方網站下載 strtus2 的完整版 (Full Distribution)。將下載到的 Zip 文件解壓縮。本文中使用的版本為 struts-2.0.11。將 struts-2.0.11 j4 文件夾下的 backport-util-concurrent-3.0.jar, retrotranslator-runtime-1.2.2.jar,struts2-core-j4-2.0.11.jar,xwork-j4-2.0.4.jar 和 lib 目錄下的 ognl-2.6.11.jar, freemarker-2.3.8.jar 文件拷貝到 portlet web 工程的 WEB-INF/lib 目錄下。
在這裡需要注意的是從 j4 文件夾下拷貝過來的 JAR 包,這是因為 WebSphere Portal Sever 5.1 基於 jdk1.4,而 j4 文件夾下的內容就是 Struts2 支持 JDK1.4 的 JAR 文件。
如果你使用的是Struts2.1.6版本,則必須將commons-fileupload-1.2.1.jar,commons-io-1.3.2.jar,commons-logging-1.0.4.jar也加入到classpath中,否則會出現無法載入Struts-default.xml文件的錯誤
生成 web.xml 配置文件
雙擊 WEB-INF/web.xml,打開 web 部署描述符界面,如圖 6 所示:
圖 6.web 部署描述符界面切換到過濾器選項卡,點擊添加按鈕,創建一個過濾器,名稱設定為 Struts2 Filter, URL 映射為 /*,並且使用現有的過濾器類 org.apache.struts2.dispatcher.FilterDispatcher,如圖 7 所示。點擊完成
圖 7. 創建一個過濾器生成的 web.xml 描述文件內容如下所示:
web.xml 描述文件內容 <?xml version="1.0" encoding="UTF-8"?><!DOCTYPE web-app PUBLIC "-//Sun Microsystems, Inc.//DTD Web Application 2.3//EN" "http://java.sun.com/dtd/web-app_2_3.dtd"><web-app id="WebApp_ID"> <display-name>Struts2TestPortlet</display-name> <filter> <filter-name>Struts2 Filter</filter-name> <display-name>Struts2 Filter</display-name> <description>Struts2 Filter</description> <filter-class> org.apache.struts2.dispatcher.FilterDispatcher </filter-class> </filter> <filter-mapping> <filter-name>Struts2 Filter</filter-name> <url-pattern>/*</url-pattern> </filter-mapping> <welcome-file-list> <welcome-file>index.html</welcome-file> <welcome-file>index.htm</welcome-file> <welcome-file>index.jsp</welcome-file> <welcome-file>default.html</welcome-file> <welcome-file>default.htm</welcome-file> <welcome-file>default.jsp</welcome-file> </welcome-file-list></web-app> |
|
生成 portlet.xml 配置文件
雙擊 portlet.xml,打開 Portlet 部署描述符界面,切換到 portlet 選項卡,如圖 8 所示:
圖 8.Portlet 部署描述符界面點擊添加按鈕,添加一個 portlet,輸入 portlet 類型 org.apache.struts2.portlet.dispatcher.Jsr168Dispatcher,確定,如圖 9 所示:
圖 9. 添加 portlet編輯該 portlet 的其它信息,分別設定 Portlet 名稱,顯示名稱,標題,簡短標題,關鍵字。本示例中均輸入 Struts2Test Portlet。如圖 10 所示:
圖 10. 編輯 portlet 信息接下來在本頁中編輯受支持的方式 text/html, 添加 portlet 模式 view,edit,help。如圖 11 所示:
圖 11. 添加 portlet 模式Portlet 模式讓 Portlet 決定它該顯示什麼內容和執行什麼動作。調用一個Portlet 的時候,Portlet 容器會提供一個 Portlet 模式給那個Portlet。當在處理一個請求動作時,Portlet 的模式是可以用程序來改變的。
JSR168 規範定義了三個 Portlet 模式:瀏覽 (View)、編輯 (Edit) 和幫助 (Help)。Struts2 支持其中的全部三個模式。同時 Portal 還可以根據使用者的角色,來決定是要提供 ( 顯示 ) 哪幾個Portlet 模式給使用者操作。在我們的例子中就使用了這三個模式。
繼續在本頁中添加初始化參數,首先是 Namespace 相關的參數見表 1:
表 1. Namespace 相關的參數名稱 | 值 |
---|---|
viewNamespace | /view |
editNamespace | /edit |
helpNamespace | /help |
考慮到在同一個 Web 應用中需要同名 Action,Struts2 以命名空間方式管理 Action。同一個命名空間里不能有同名的Action,不同的命名空間里可以有同名的 Action。Struts2 不支持為單獨的 Action 設置命名空間,而是通過為包指定namespace 屬性來為下面所有的 Action 指定共同的命名空間。這一點可以從下面的 portlet.xml清單中看出。Struts2 對 portlet 的三種模式的支持是通過 namespace 體現的。View, Edit, Help三種模式分別對應 ViewNamespace, editNamespace, helpNamespace。
Action 相關的參數見表 2:
表 2. Action 相關的參數名稱 | 值 |
---|---|
defaultViewAction | view |
defaultEditAction | edit |
defaultHelpAction | help |
這些參數指定了在 portlet 的三種模式下的默認 aciton 名稱,進入 View、Edit、Help 模式分別首先調用名稱為 view、edit、help 的 action。
圖 12. 編輯 portlet 初始化參數生成 portlet.xml 文件如下:
portlet.xml 描述文件內容 <?xml version="1.0" encoding="UTF-8"?><portlet-app xmlns="http://java.sun.com/xml/ns/portlet/portlet-app_1_0.xsd" version="1.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://java.sun.com/xml/ns/portlet/portlet-app_1_0.xsd http://java.sun.com/xml/ns/portlet/portlet-app_1_0.xsd" id="com.ibm.faces.portlet.FacesPortlet.3ccbdcb861"> <portlet> <portlet-name>Struts2 Test Portlet</portlet-name> <display-name>Struts2 Test Portlet</display-name> <portlet-class> org.apache.struts2.portlet.dispatcher.Jsr168Dispatcher </portlet-class> <init-param> <name>viewNamespace</name> <value>/view</value> </init-param> <init-param> <name>editNamespace</name> <value>/edit</value> </init-param> <init-param> <name>helpNamespace</name> <value>/help</value> </init-param> <init-param> <name>defaultViewAction</name> <value>view</value> </init-param> <init-param> <name>defaultEditAction</name> <value>edit</value> </init-param> <init-param> <name>defaultHelpAction</name> <value>help</value> </init-param> <supports> <mime-type>text/html</mime-type> <portlet-mode>view</portlet-mode> <portlet-mode>edit</portlet-mode> <portlet-mode>help</portlet-mode> </supports> <portlet-info> <title>Struts2 Test Portlet</title> <short-title>Struts2 Test Portlet</short-title> <keywords>Struts2 Test Portlet</keywords> </portlet-info> </portlet></portlet-app> |
在本清單中 Portlet類「org.apache.struts2.portlet.dispatcher.Jsr168Dispatcher」在將 Struts2 集成到Portlet 中起到了關鍵作用,該類將 Portlet 操作分發給 Struts2。
|
編輯 jsp 文件
在 WEB-INF 下新建 jsp 文件夾,並生成幾個 jsp 文件及內容如下 :
input.jsp <%@ taglib prefix="s" uri="/struts-tags"%><H2>Please sign in</H2><s:form action="login" method="POST"> <s:textfield label="User Name" name="username" value="%{username}" /> <s:password label="Password" name="password" value="%{password}" /> <s:submit value="Sign in" /></s:form> |
success.jsp
<%@ taglib prefix="s" uri="/struts-tags"%><H2>Welcome <s:property value="username" /></H2><p /><a href="<s:url action="view"/>">Back to sign in page</a> |
fail.jsp
<%@ taglib prefix="s" uri="/struts-tags"%><H2>Invalid username or password!</H2><p /><a href="<s:url action="view"/>">Back to sign in page</a> |
edit.jsp
<%@ taglib prefix="s" uri="/struts-tags"%><H2>This is edit page!</H2> |
help.jsp
<%@ taglib prefix="s" uri="/struts-tags"%><H2>This is help page!</H2> |
|
應用 Struts2
建立 Struts2 配置文件 struts.xml
在 src 目錄下建立 struts.xml,工程構建之後 struts.xml 會被置於 WEB-INF/classes 下,這也是 Struts2 默認的配置文件路徑。
struts.xml 文件是基於 Struts2 的項目的核心配置文件。在 portlet 開發過程最重要的內容就是要將"struts-portlet-default.xml" 包含進來,這個 xml 文件存在於前面步驟中我們拷貝過來的struts2-core-j4-2.0.11.jar 中,只有把它包含進來我們才能使 Struts2 支持portlet。接下來描述的內容就是將前面定義的 action 和頁面關聯起來。從中可以看出我們定義的 login 這個 actoin的輸入界面是 input.jsp,如果成功則會被轉向 success.jsp,失敗則會被轉向 fail.jsp。
struts.xml <?xml version="1.0" encoding="UTF-8" ?><!DOCTYPE struts PUBLIC "-//Apache Software Foundation//DTD Struts Configuration 2.0//EN" "http://struts.apache.org/dtds/struts-2.0.dtd"><struts> <include file="struts-portlet-default.xml" /> <package name="view" extends="struts-portlet-default" namespace="/view"> <action name="view"> <result>/WEB-INF/jsp/input.jsp</result> </action> <action name="login" class="struts2TestPortlet.action.Login"> <result name="input">/WEB-INF/jsp/input.jsp</result> <result name="success">/WEB-INF/jsp/success.jsp</result> <result name="fail">/WEB-INF/jsp/fail.jsp</result> </action> </package> <package name="edit" extends="struts-portlet-default" namespace="/edit"> <action name="edit"> <result>/WEB-INF/jsp/edit.jsp</result> </action> </package> <package name="help" extends="struts-portlet-default" namespace="/help"> <action name="help"> <result>/WEB-INF/jsp/help.jsp</result> </action> </package></struts> |
新建 java 類 Login.java
Login.java package struts2TestPortlet.action;public class Login extends ActionSupport { String username; String password; public String getPassword() { return password; } public void setPassword(String password) { this.password = password; } public String getUsername() { return username; } public void setUsername(String username) { this.username = username; } public String execute() throws Exception { if (username.equals("yanzhid") && password.equals("pwd")) { return "success"; } else { return "fail"; } }} |
同 Struts1 的比較
在實現相同功能的前提下,我們不妨看一下 Struts 1 的相關實現,示例中採用的是 IBM 的 Struts forJSR168 Portlet 實現。
首先,我們必須針對用戶輸入的表單數據創建一個 ActionForm,當然,您也可以通過 xml 文件配置的方式創建一個 DynamicForm。
java 類 LoginActionForm.java public class LoginActionForm extends ActionForm { String username; String password; public String getPassword() { return password; } public void setPassword(String password) { this.password = password; } public String getUsername() { return username; } public void setUsername(String username) { this.username = username; } public void reset(ActionMapping mapping, HttpServletRequest request) { username = null; password = null; } public ActionErrors validate(ActionMapping mapping, HttpServletRequest request) { ActionErrors errors = new ActionErrors(); return errors; }} |
接下來,我們需要實現一個 Action 類。
java 類 LoginAction.java public class LoginAction extends com.ibm.portal.struts.action.StrutsAction { public ActionForward execute(ActionMapping mapping, ActionForm form, PortletRequest request, PortletResponse response) throws Exception {|-------10--------20--------30--------40--------50--------60--------70--------80--------9||-------- XML error: The previous line is longer than the max of 90 characters ---------| LoginActionForm loginActionForm = (LoginActionForm) form; String username = loginActionForm.getUsername(); String password = loginActionForm.getPassword(); if (username.equals("yanzhid") && password.equals("pwd")) { StrutsViewCommand.addAttributeNameToSave("username"); request.setAttribute("username", username); return mapping.findForward("success"); } else { return mapping.findForward("fail"); } }} |
在 Struts1 中,我們習慣使用 JSTL 從 request 作用域中取得數據,呈現到 JSP 頁面上,success.jsp 頁面被改動如下:
success.jsp <%@page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%><%@taglib uri="http://java.sun.com/portlet" prefix="portlet"%><%@taglib uri="http://java.sun.com/jstl/core" prefix="c"%><portlet:defineObjects /><H2>Welcome <c:out value="${username}"/></H2> |
由於精通 Struts1 的讀者很多,其餘相關改動和 struts1 的配置我們不再拗述,請讀者查閱相關文獻。
從中我們可以看出 struts2 和 struts1 的區別,並從中體會到 Struts2 的優勢:
- 捕獲輸入: Struts1 使用 ActionForm 對象捕獲輸入。所有的 ActionForm 必須繼承一個基類。因為其他 JavaBean 不能用作 ActionForm,開發者經常創建多餘的類捕獲輸入。當然您也可以使用 xml 文件描述一個 DynamicForm,但工作量並未顯著減少。 Struts 2 可以直接使用 Action 屬性作為輸入屬性,消除了對第二個輸入對象的需求。當然,Struts2 也支持 ActionForm 模式,用作輸入 / 輸出對象。
- Action 類 : Struts1 要求 Action 類繼承一個抽象基類。Struts1 的一個普遍問題是使用抽象類編程而不是介面。 Struts2 Action 類可以實現一個 Action 介面,也可實現其他介面。Struts2 提供一個 ActionSupport 基類去實現常用的介面,但 Action 介面不是必須的,任何有 execute 標識的 POJO 對象都可以用作 Struts2 的 Action 對象。
- Portlet 依賴 : Struts1 Action 依賴於 Portlet API , 當一個 Action 被調用時,PorletRequest 和 PortletResponse 被傳遞給 execute 方法。 Struts2 Action 不依賴於容器,允許 Action 脫離容器單獨被測試。如果需要,Struts2 Action 仍然可以訪問初始的 request 和 response。但是,其他的元素減少或者消除了直接訪問 PorletRequest 和 PortletResponse 的必要性。
- 可測性 : 測試 Struts1 Action 的一個主要問題是 execute 方法暴露了 Portlet API,這使得測試要依賴於容器。 Struts2 Action 可以通過初始化、設置屬性、調用方法來測試,使測試更容易。
- 對 Portlet 的支持程度: Struts1 框架天生並不支持 Portlet, 為了將大批 Struts 程序員吸引到 WebSphere Portal 開發上來,IBM 公司針對自己的兩種 Portlet 實現(IBM Portlet 和 JSR168 Portlet), 分別對 Struts 框架進行了擴展,以適應 Portlet 開發。細心的讀者一定已經發現了,我們的 LoginAction 類繼承的是 com.ibm.portal.struts.action.StrutsAction 這個基類,而不是 Struts1 官方框架中的 Action 類。 Struts1 程序員剛剛進行 Portlet 開發的時候,一般會忽略一個重要的問題。與 servlet 的生命周期有所不同,portlet 存在操作響應 (Action) 階段和呈現(Render)階段。 在 portlet 操作響應階段存放在 request 作用域的數據,在呈現階段就會失效。為了解決這個問題,一個方案就是把數據放在 session 作用域中,考慮到伺服器性能的原因,這個方案顯然不被推薦。另一個方案就是使用 IBM 提供的一些特殊的 API, 按照 LoginAction 那樣,在將數據放在 request 作用域之前,將數據變數名作為參數傳給 StrutsViewCommand 的 addAttributeNameToSave 方法。為了說明這個問題,我們可以將 StrutsViewCommand.addAttributeNameToSave("username") 這行代碼刪掉,程序運行候就可以發現,success.jsp 將不會呈現 username 變數中保存的數據。 Struts2 的官方框架支持 JSR168 Portlet, 除了前期需要對 web.xml、portlet.xml 和 struts.xml 文件進行少許特殊配置外,程序員不需要考慮 Portlet 和 Servlet 之間的不同,無論是 Action 類的開發還是 JSP 頁面中 Struts2 標籤的應用,同在 Servlet 容器中用法是完全一樣的。
Struts2 的驗證功能
採用 Struts2 的校驗框架時,只需為該 Action 指定一個校驗文件即可。該文件指定了 Action的屬性必須滿足怎樣的規則,下面清單即本應用中 Action 的校驗文件的代碼。Struts2 的校驗文件規則與 Struts1的校驗文件設計方式不同,Struts2 中每個 Action 都可以有一個校驗文件,該文件的文件名遵守如下規則:
<Action 名字 >-validation.xml
前面的 Action 名是可以改變的,後面的 -validation.xml 部分是固定不變的,且該文件應該被保存於 Actionclass文件相同的路徑下。如在本例中此校驗文件在項目構建後保存在 WEB-INF/classes/Struts2TestPortlet/action/ 下,與 login.class 在同一目錄中。
Login-validation.xml <?xml version="1.0" encoding="UTF-8" ?><!DOCTYPE validators PUBLIC "-//OpenSymphony Group//XWork Validator 1.0.2//EN" "http://www.opensymphony.com/xwork/xwork-validator-1.0.2.dtd"><validators> <field name="username"> <field-validator type="requiredstring"> <param name="trim">true</param> <message>Please input username!</message> </field-validator> </field> <field name="password"> <field-validator type="requiredstring"> <param name="trim">true</param> <message>Please input password!</message> </field-validator> </field></validators> |
|
部署 Portlet 應用程序
首先在 RAD7 控制台的伺服器視圖中添加一個可以運行 Portlet 的伺服器。該視圖中單擊滑鼠右鍵,選擇新建—> 伺服器。如圖 13 所示 :
圖 13. 新建伺服器在彈出的新建伺服器窗口中選擇目標運行伺服器,在本例中我們選擇 WebSphere PortalV5.1 測試環境 , 對話框底部的伺服器運行時呈現為 WebSphere PortalV5.1。在伺服器主機名中輸入 localhost,如圖 14 所示。點擊下一步,默認埠號為 9081。
圖 14. 選擇目標運行伺服器繼續點擊下一步,出現門戶網站伺服器設置對話框,在這裡需要將默認的管理員用戶及密碼設置成您在安裝相應 Portalserver 時設置的管理員用戶及密碼。如圖 15 所示:
圖 15. 門戶網站伺服器設置點擊下一步,進入添加項目界面,將 " 可用的項目 " 一欄中的 Portlet項目添加到右側 " 已配置項目 " 欄中。點擊完成,於是我們就完成了這個 Portlet 項目在 RAD7 中的配置和部署。如圖 16 所示:
圖 16. 添加 portlet 項目
在啟動伺服器之前,需要右鍵點擊伺服器標籤頁中的已添加的伺服器,選擇 " 打開 ",然後進入 " 門戶網站 " 標籤頁,檢查確認管理員用戶和密碼已經被設置成安裝 Portalserver 時設置的管理員用戶,如圖 17 所示。以確保 Portal server 啟動成功。
圖 17. 檢查確認管理員用戶和密碼設置接下來,右鍵點擊伺服器標籤頁中已添加的伺服器,選擇 " 啟動 ",伺服器被啟動。您可以從控制台中觀察啟動過程輸出的日誌。
圖 18. 啟動伺服器
|
訪問 Portlet 應用
打開瀏覽器網頁輸入:http://localhost:9081/wps/myportal, 輸入管理員用戶和密碼,進入 Portal。程序轉入 portletview 模式的登錄界面。如圖 19 所示:
圖 19. 登錄界面該 portlet 中用戶需要輸入自己的用戶名和密碼,本例中用戶名和密碼分別為 yanzhid 和 pwd,輸入後點擊 signin 後進入成功頁面,如圖 20 所示:
圖 20. 登錄成功界面如果用戶未輸入合法的用戶名和密碼,程序將跳轉到登錄失敗頁面。如圖 21 所示:
圖 21. 登錄失敗界面通過點擊 edit 按鈕,potrlet 進入 edit 模式,如圖 22 所示:
圖 22.edit 模式通過點擊 help 按鈕,potrlet 調用 help 模式,彈出 help 窗口,如圖 23 所示:
圖 23.help 模式為了使用本例中的 Struts2 的校驗功能,返回到登錄頁面,重新登錄。登錄時不輸入任何用戶名和密碼,直接點擊 "Signin" 按鈕,您會看到用戶名和密碼文本框上方出現 "Please input username!" 和 "Pleaseinput password!" 字樣,則說明本例中的校驗功能正確執行了。如圖 24 所示:
圖 24.Struts2 的校驗功能
|
結束語
本文討論了如何使用 Struts2 在 IBM WebSpherePortal 上來開發和部署一個 Portlet 應用,按照本文講述的步驟操作,就可以在 WebSherePortal 上建立一個比較完整的 Portlet 應用了。
推薦閱讀:
※Struts標籤庫
※Struts2 初探
※Struts2學習第一天
※Struts 2 中 <s:form> 標籤和 HTML 的 <form> 標籤有什麼區別?
※JSP中標籤的部署與調用