译者:holexp
概要
在stefan hepper和stephan hesmer的portlet系列文章的第二部分中,作者把着笔点从portlet api的基础概要介绍转移到了portlet api的参考实现(ri reference implementation也就是pluto)的细节描述。作者还提供了一系列portlet的实例来说明怎样扩展portlet api的标准函数。
企业portal提供商使用可插的用户接口组件(portlets)向信息系统提供表示层。不幸的是,以前的提供商都只定义了自己的portlet api,在整个行业之中互不相容。为了标准化整个行业进程,java团体发布了java规范要求(jsr)168:portlet规范。
这篇系列文章的第一部分介绍了jsp 168的细节。第二部分重点放在portlet api的参考实现(ri)上,也就是pluto。此外还提供了一个portlet的实例,读者可以通过这个实例来学习。
文章第一节描述了ri的体系结构,包括portlet容器的可拆卸性的概念和怎样在其他项目中重用portlet容器。第二节介绍了ri的安装和使用,以及怎样快速配置portlet。其中文章还包括一个逐步深入的实例。
注意:你可以通过文章之后的资源链接下载原代码
pluto的体系结构
让我们先来看一下pluto的体系结构和一些基本的概念。我们先简要的说明portal的参考实现和portlet容器在整个portal体系结构中的位置。接下来我们在细节方面研究pluto的体系结构。最后,我们看一下在portlet容器里很有趣的:portlet 展开。
关于portal
pluto一般用来演示portlet api如何工作以及向开发者提供一个测试portlets的实例平台。然而,如果没有驱动来运行和测试portlet容器有点麻烦。pluto的简单portal组件只是架构于portlet容器,它只满足了jsr 168的基本要求。(相比之下,apache的开源项目jetspeed就要专业的多。jetspeed将着重中在了portal本身而非portlet容器之上,并且更多的考虑了其他团体的需求。)
图一描述了portal的基本体系结构。portal的网络应用程序处理客户端请求,从用户的当前页面得到portlets,之后调用portlet容器以获得每个portlet的内容。portal使用portlet 容器的 invoker api来访问 portlet容器,从 portal看来,portlet 容器的主要接口是支持基于请求的方法调用 portlets。容器用户要想获得portal的相关信息则必须实现portlet容器的provider spi (service provider interface)的callback接口。最终,portlet容器通过portlet api调用所有portlet。
图一:pluto中的一个简单的portal的结构
portlet容器
portlet容器是portlet的运行环境,也是每个portal的组成核心。它需要有关portal本身的信息,且它必须重用自身的公共代码。因此,portlet容器和其他portal组件是完全分离的。这就是说,你可以将独立的portlet容器嵌入任意的portal,只要你满足portlet容器的条件,比方说实现所有的spi。
portlet容器的 invoker api,或者叫入口点,扮演了portlet容器的主调用接口的角色。portlet容器的invoker api将portlet容器的生存周期(init,destroy)和基于请求的调用方法(initpage(),performtitle(),portletservice()等等)结合了起来。因为portlet容器最后调用portlet的方法名有点类似portlet api的主portlet接口,不同的是是否必须要传递portlet定义符。正是因为这个附加的portlet定义符,portlet容器才能正确的调用portlet。
除了要用api访问portlet容器之外,portal还必须扩展portlet容器定义的spi。因此,ri引入了容器服务:在容器注册过的可拆卸组件提供基础功能并且可扩充。ri包含如下一些容器内的自建服务(前四个必须在运行portlet容器时实现,最后一个是可选的):
 信息提供器:给portlet容器提供portal和portal框架的信息。通过这个接口来获得信息和存储portal信息。这些信息包括导航栏里的url、portlet上下文、portlet模式和窗口状态控制。
工厂管理器:定义了怎么怎样通过工厂方法来获得一个具体实现。(一个标准的portal应该已经存在一个实现。)
 日志服务:定义了一个日志工具(一个标准的portal应该已经存在一个实现)。
 配置服务:定义了怎么样获得配置参数(一个标准的portal应该已经存在一个实现)。
 属性管理器(可选):属性管理器接口的实现允许处理jsr168规范中定义的属性。
严格的说,portlet对象模型也是spi中的一部分,只是它在spi中占有一个特殊的地位。portlet对象模型处理所有的potlet对象,他由一个交织在一起的接口集合组成。因此,不能把他和容器服务分开来考虑。
图二:portlet容器结构
portlet的部署
portlet 容器 架构在servlet容器之上并且增强了它的功能。为了实现它,portlet 容器将原始servlet 加入每一个portlet应用程序的war文件中,这一点我们在图三3中有所描述。部署portlet组件时,先取得原始的war文件,然后向其中加入一个新的或者修改原有的web.xml,并且加入一个servlet作为一个调用点来包装每个portlet。之后, portlet的部署器(?这个原文是then the portlet deployment passes the modified war file to the application server deployment)会传递一个修改过的war文件到应用服务器,将其部署到应用服务器系统。在portlet的调用过程中,portlet容器调用添加进去的servlet,作为部署portlet的war文件的入口点。
图三:ri中portlet的部署
pluto和wsrp标准
正像第一部分所描述的那样,jsr 168与远程portlet网络服务(the web services for remote portlets (wsrp))标准紧密结合。几乎同时形成的这两种标准发布了开源实现,实现了在各自的规范中描述必要的功能。作为共有的目标,两种标准努力能够在一起更好的合作。现在,portlet容器可以很好的运行wsrp portlet。
pluto可以在一个portal中运行多个portlet容器。从而pluto的portlet容器可以被初始化多次。更重要的是,可以用不同的方式来初始化它。每一个portlet容器可以使用spi的不同实现。
ri的安装
你会发现pluto的安装过程非常简单。执行install命令,build目录/build下的install.bat或者install.sh。接下来安装程序会提示你指定tomcat的安装目录。(注意:在ms windows下文件分隔符不是反斜杠。)
在这之后,安装进程会创建ri和所有portlet,安装portlet到指定的tomcat目录。安装完成后请查看文档以确定完成了所有必要的手工设置工作。
现在可以启动tomcat,通过http://localhost:8080/pluto/portal来访问ri了。
就是这么简单!
怎样部署portlet
在pluto中部署portlet和它的安装一样的简单。只要记住你必须首先安装了pluto,它正确的设置了preparerun.properties。这是部署过程所必须的。在命令提示符下转到build目录,输入命令deployportlet.bat , 用portlet war文件做参数,比如:
deployportlet.bat c:\pluto\portlets\bookmark_04\driver\bookmark_04.war
portlet实例
我们来看一个portlet的例子,bookmark。它充分利用了portlet api并且阐明了我们学到的概念。我们以一个简单的例子开始,我们在每一节一步步扩展这个bookmark portlet,最后我们将几乎用到所有的portlet api,把它做成一个高级的portlet。
bookmark portlet:版本一
第一个bookmark portlet用到了portlet api中如下的一些特性:
 portlet api 接口the portlet api interface
 java服务器页面(jsp)javaserver pages (jsp) pages
 portlet api标签库the portlet api tag libraries
 部署描述符deployment descriptors
第一个bookmark portlet的两个jsp页面分别显示和编辑模式。每个jsp页面只是简单的显示了portlet的当前portlet模式和windwos状态。为了显示这些信息,我们用到了portlet api标签库(只是部分程序代码,请下载全部代码,不然很难理解:译者注):
public void doview (renderrequest request,
renderresponse response) throws portletexception, ioexception {
response.setcontenttype("text/html");
string jspname = getportletconfig().getinitparameter("jspview");
portletrequestdispatcher rd =
getportletcontext().getrequestdispatcher(jspname);
rd.include(request,response);
}
<%@ page session="false" %>
<%@ page import="javax.portlet.*"%>
<%@ page import="java.util.*"%>
<%@ taglib uri='/web-inf/tld/portlet.tld' prefix='portlet'%>
<portlet:defineobjects/>
hello,<br>
i am the bookmark portlet.<br>
<br>
current portlet mode: <%=portletrequest.getportletmode()%><br>
current window state: <%=portletrequest.getwindowstate()%><br>
<br>
Java Asp PHP .Net XML C/C++ CGI VB Jsp J2ee J2se J2me EJB Servlet Tomcat Resin Struts Weblogic Eclipse ANT GUI JMS Web servise IDEA Webphere Hibernate Spring Jboss Applet Swing Socket Javamail Perl Ajax P2P 安全 模式 框架 测试 开源 游戏
Windows XP Windows 2000 Windows 2003 Windows Me Windows 9.x Linux UNIX 注册表 操作系统 服务器 应用服务器