WebSphere Portal Server5.1 及以上版本支持两种 Portlet API:第一种是 IBM Portlet,这种 Portlet API 是 WebSphere Portal Server 专有的一种 Portlet API;第二种是符合 JSR168 标准的 Portlet API。由于 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 怎样解决旧有的 Portlet API 所无法克服的困难的。
本文的重点不在于开发一个 Struts2 Web 应用程序,而在于开发一个作为 Portlet 的 Struts2 应用程序所需的的实现和配置。读者可以了解到如何利用 Struts 2 来创建一个 Portlet,这个 Portlet 将完全独立于其所开发和部署的平台。
在示例应用程序的开发和部署中用到了下列产品:
示例应用程序是一个简单的用户登录程序。合法的用户将跳转到的登录成功页面,登录失败的用户则跳转到登录失败页面,并被要求输入正确的用户名和密码。用户可以自由的在 Portlet 的 View、Edit 和 Help 模式之间进行切换。应用程序视图部分分为以下几部分:
下图是本文示例的 Action 与页面的交互图:
在本文中,使用 Struts2 开发 Portlet 应用需要经历以下步骤:
在 Rational Application developer 7 中启动一个 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 文件。
双击 WEB-INF/web.xml,打开 web 部署描述符界面,如图 6 所示:
切换到过滤器选项卡,点击添加按钮,创建一个过滤器,名称设定为 Struts2 Filter, URL 映射为 /*, 并且使用现有的过滤器类 org.apache.struts2.dispatcher.FilterDispatcher,如图 7 所示。点击完成
生成的 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 部署描述符界面,切换到 portlet 选项卡,如图 8 所示:
点击添加按钮,添加一个 portlet, 输入 portlet 类型 org.apache.struts2.portlet.dispatcher.Jsr168Dispatcher,确定,如图 9 所示:
编辑该 portlet 的其它信息,分别设定 Portlet 名称,显示名称,标题,简短标题,关键字。本示例中均输入 Struts2 Test Portlet。如图 10 所示:
接下来在本页中编辑受支持的方式 text/html, 添加 portlet 模式 view,edit, help。如图 11 所示:
Portlet 模式让 Portlet 决定它该显示什么内容和执行什么动作。调用一个 Portlet 的时候,Portlet 容器会提供一个 Portlet 模式给那个 Portlet。当在处理一个请求动作时,Portlet 的模式是可以用程序来改变的。
JSR168 规范定义了三个 Portlet 模式: 浏览 (View)、编辑 (Edit) 和帮助 (Help)。Struts2 支持其中的全部三个模式。同时 Portal 还可以根据使用者的角色,来决定是要提供 ( 显示 ) 哪几个 Portlet 模式给使用者操作。在我们的例子中就使用了这三个模式。
继续在本页中添加初始化参数,首先是 Namespace 相关的参数见表 1:
/view |
/edit |
/help |
考虑到在同一个 Web 应用中需要同名 Action,Struts2 以命名空间方式管理 Action。同一个命名空间里不能有同名的 Action,不同的命名空间里可以有同名的 Action。Struts2 不支持为单独的 Action 设置命名空间,而是通过为包指定 namespace 属性来为下面所有的 Action 指定共同的命名空间。这一点可以从下面的 portlet.xml 清单中看出。Struts2 对 portlet 的三种模式的支持是通过 namespace 体现的。View, Edit, Help 三种模式分别对应 ViewNamespace, editNamespace, helpNamespace。
Action 相关的参数见表 2:
view |
edit |
help |
这些参数指定了在 portlet 的三种模式下的默认 aciton 名称,进入 View、Edit、Help 模式分别首先调用名称为 view、edit、help 的 action。
生成 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。
在 WEB-INF 下新建 jsp 文件夹,并生成几个 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> |
<%@ 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> |
<%@ 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> |
<%@ taglib prefix="s" uri="/struts-tags"%> <H2>This is edit page!</H2> |
<%@ taglib prefix="s" uri="/struts-tags"%> <H2>This is help page!</H2> |
在 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。
<?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> |
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"; } } } |
在实现相同功能的前提下,我们不妨看一下 Struts 1 的相关实现,示例中采用的是 IBM 的 Struts for JSR168 Portlet 实现。
首先,我们必须针对用户输入的表单数据创建一个 ActionForm,当然,您也可以通过 xml 文件配置的方式创建一个 DynamicForm。
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 类。
public class LoginAction extends com.ibm.portal.struts.action.StrutsAction { public ActionForward execute(ActionMapping mapping, ActionForm form, PortletRequest request, PortletResponse response) throws Exception { 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 页面被改动如下:
<%@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 的优势:
采用 Struts2 的校验框架时,只需为该 Action 指定一个校验文件即可。该文件指定了 Action 的属性必须满足怎样的规则,下面清单即本应用中 Action 的校验文件的代码。Struts2 的校验文件规则与 Struts1 的校验文件设计方式不同,Struts2 中每个 Action 都可以有一个校验文件,该文件的文件名遵守如下规则:
<Action 名字 >-validation.xml
前面的 Action 名是可以改变的,后面的 -validation.xml 部分是固定不变的,且该文件应该被保存于 Action class 文件相同的路径下。如在本例中此校验文件在项目构建后保存在 WEB-INF/classes/Struts2TestPortlet/action/ 下,与 login.class 在同一目录中。
<?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> |
首先在 RAD7 控制台的服务器视图中添加一个可以运行 Portlet 的服务器。该视图中单击鼠标右键,选择新建—> 服务器。如图 13 所示 :
在弹出的新建服务器窗口中选择目标运行服务器,在本例中我们选择 WebSphere Portal V5.1 测试环境 , 对话框底部的服务器运行时呈现为 WebSphere Portal V5.1。在服务器主机名中输入 localhost,如图 14 所示。点击下一步,默认端口号为 9081。
继续点击下一步,出现门户网站服务器设置对话框,在这里需要将默认的管理员用户及密码设置成您在安装相应 Portal server 时设置的管理员用户及密码。如图 15 所示:
点击下一步,进入添加项目界面,将 " 可用的项目 " 一栏中的 Portlet 项目添加到右侧 " 已配置项目 " 栏中。点击完成,于是我们就完成了这个 Portlet 项目在 RAD7 中的配置和部署。如图 16 所示:
在启动服务器之前,需要右键点击服务器标签页中的已添加的服务器,选择 " 打开 ",然后进入 " 门户网站 " 标签页,检查确认管理员用户和密码已经被设置成安装 Portal server 时设置的管理员用户,如图 17 所示。以确保 Portal server 启动成功。
接下来,右键点击服务器标签页中已添加的服务器,选择 " 启动 ",服务器被启动。您可以从控制台中观察启动过程输出的日志。
打开浏览器网页输入:http://localhost:9081/wps/myportal, 输入管理员用户和密码,进入 Portal。程序转入 portlet view 模式的登录界面。如图 19 所示:
该 portlet 中用户需要输入自己的用户名和密码,本例中用户名和密码分别为 yanzhid 和 pwd,输入后点击 sign in 后进入成功页面,如图 20 所示:
如果用户未输入合法的用户名和密码,程序将跳转到登录失败页面。如图 21 所示:
通过点击 edit 按钮,potrlet 进入 edit 模式,如图 22 所示:
通过点击 help 按钮,potrlet 调用 help 模式,弹出 help 窗口,如图 23 所示:
为了使用本例中的 Struts2 的校验功能,返回到登录页面,重新登录。登录时不输入任何用户名和密码,直接点击 "Sign in" 按钮,您会看到用户名和密码文本框上方出现 "Please input username!" 和 "Please input password!" 字样,则说明本例中的校验功能正确执行了。如图 24 所示:
本文讨论了如何使用 Struts2 在 IBM WebSphere Portal 上来开发和部署一个 Portlet 应用,按照本文讲述的步骤操作,就可以在 WebShere Portal 上建立一个比较完整的 Portlet 应用了。