IBM中國軟件實(shí)驗(yàn)室 2004 年 7 月
本文介紹了最新版WSAD 5.1.2上基于 JSF技術(shù)的Faces Portlet 框架的特色,從 MVC模式角度與基本的 portlet 進(jìn)行了比較,并進(jìn)一步深入分析了Faces portlet 開發(fā)過程中的關(guān)鍵類的具體含義與功能。
簡介 最新發(fā)布的 IBM WebSphere Studio Application Developer 和 IBM WebSphere Studio Site Developer 5.1.2 版支持新的行業(yè)標(biāo)準(zhǔn),可以簡化針對豐富 Web 用戶界面、商務(wù)邏輯(business logic)和交互式門戶的開發(fā)。全新的快速應(yīng)用程序開發(fā)(RAD)和代碼生成工具是WebSphere Studio 的一大亮點(diǎn),可以幫助實(shí)現(xiàn)項(xiàng)目流程自動化,簡化Java開發(fā),從而加快整個團(tuán)隊(duì)的開發(fā)速度。
使用新的WebSphere Studio產(chǎn)品來構(gòu)建豐富的用戶界面、數(shù)據(jù)連接以及面向Web應(yīng)用程序的商務(wù)邏輯,Java開發(fā)人員可以立即獲得工作效率的提升。WebSphere Studio 支持最近獲得Java Community ProcessSM批準(zhǔn)的 JavaServer? Faces (JSF)標(biāo)準(zhǔn)和Java Community ProcessSM推薦的Service Data Objects(SDO)標(biāo)準(zhǔn)。WebSphere Studio使用這些標(biāo)準(zhǔn)來最大限度地減少構(gòu)建數(shù)據(jù)驅(qū)動的Web應(yīng)用程序和豐富的Web用戶界面所需要的Java編碼和手動操作。JSF和SDO的可視化工具使開發(fā)人員可以方便地將JSF用戶界面(UI)組件拖放到頁面,點(diǎn)擊即可連接到關(guān)系型數(shù)據(jù)庫的數(shù)據(jù)源。
因此,借助WSAD V5.1.2開發(fā)工具的幫助,我們可以使用JSF相關(guān)的技術(shù)來實(shí)現(xiàn)終端用戶自己開發(fā)的小型系統(tǒng),更大規(guī)模的門戶應(yīng)用,或者是規(guī)模龐大的企業(yè)系統(tǒng)。
此外,該版本的WebSphere Studio Application Developer 還包括 IBM WebSphere Portal 測試環(huán)境,新的可視化 portlet 開發(fā)工具,并支持新行業(yè)標(biāo)準(zhǔn) portlet Application Programming Interface (API),可確保 portlet 互操作性和便攜性。構(gòu)建 portlet 的可視化工具可簡化門戶開發(fā),而與 JSF 的集成使其可以輕易地整合 portlet 中的豐富的用戶界面(UI)組件和 Web 窗體。與過去相比,這大大簡化了整個門戶開發(fā)過程,速度也大幅提高,但對相應(yīng)開發(fā)人員技能卻沒有很高的要求。 這就是最新的Faces Portlet開發(fā)框架。
本文是有關(guān)Faces Portlet開發(fā)的系列文章的第一篇,在這些文章中作者會和開發(fā)人員談?wù)撚嘘P(guān)Faces Portlet開發(fā)方面很多有趣的話題。
目標(biāo)讀者:希望讀者閱讀本文之前,能夠?qū)VC,portlet和JSF有一定的了解,相關(guān)的資料可以在參考資料中找到。
出發(fā)之前,先來準(zhǔn)備我們的開發(fā)測試環(huán)境吧。
Faces Portlet開發(fā)測試環(huán)境的準(zhǔn)備:我們使用的開發(fā)工具是WSAD V5.1.2,同時還需要安裝IBM提供的Portal Toolkit V5.0.2.2。
- WSAD V5.1.2
您可以從下面這個鏈接下載到WSAD V5.1.2的試用版,請 下載安裝。注意:這里一般不能從WSAD的老版本通過更新管理器來升級您的WSAD版本,最方便可靠的辦法是重新安裝這個新的版本。
- 運(yùn)行環(huán)境:
WebSphere Portal V5.0.2或者WebSphere Portal Express V5.0.2 都可以用來發(fā)布Faces Portlet應(yīng)用。
好了,做完上面這一切,你就可以開始著手開發(fā)Faces Portlet應(yīng)用了。
(圖1)
如上圖(圖1),WSAD V5.1.2提供了四種不同的創(chuàng)建portlet應(yīng)用的向?qū)Аi_發(fā)人員以前最常用的是Basic portlet這個向?qū)?,這里我們就先來比較一下Basic portlet和Faces portlet有哪些區(qū)別和類似之處。
傳統(tǒng)上Basic portlet的開發(fā)過程 http://www.ibm.com/developerworks/cn/websphere/library/techarticles/0403_lynn/0403_lynn.html從上面這篇文章,你可以了解到以前開發(fā)一個最簡單的portlet的一般步驟,最基本的步驟如下:
1. 創(chuàng)建目錄結(jié)構(gòu):使用Basic portlet的創(chuàng)建向?qū)?,WSAD會幫你自動創(chuàng)建目錄結(jié)構(gòu)。
2. 準(zhǔn)備JSP文件: 準(zhǔn)備好需要顯示的所有JSP頁面。
3. 創(chuàng)建Portlet: 創(chuàng)建Portlet類,它通常是繼承于PortletAdapter的,需要實(shí)現(xiàn)的最重要的方法是actionPerformed()和doView()。
4. 準(zhǔn)備配置文件: 打包發(fā)布之前需要維護(hù)web.xml以及portlet.xml,通常WSAD會幫你完成這一切。
5. 打包,發(fā)布這個應(yīng)用:將你的應(yīng)用導(dǎo)出成WAR包,然后在WebSphere Portal Server上安裝,并且把它裝載到某個頁面中,你就可以看到它的外觀了。
Faces Portlet的主要優(yōu)點(diǎn) Faces portlet脫胎于Basic portlet,并在其基礎(chǔ)上將JSF技術(shù)整合進(jìn)來,充分吸收了JSF的很多優(yōu)點(diǎn),包括豐富的用戶界面控件,便捷的事件處理和頁面導(dǎo)航等特性,進(jìn)一步降低了開發(fā)portlet應(yīng)用的門檻。如果能夠充分利用WSAD V5.1.2提供的工具支持的話,開發(fā)人員幾乎感覺不到是在開發(fā)portlet應(yīng)用,而只是在開發(fā)基于JSF技術(shù)的動態(tài)Web應(yīng)用。
正是由于項(xiàng)目中有了下面這些Jar包的支持,F(xiàn)aces portlet才能正常工作。
jsf-api.jar,jsf-ibm.jar,jsf-impl.jar,jsf-portlet.jar
下面,我們再來從實(shí)現(xiàn)MVC模式的角度來比較Basic portlet和Faces portlet的異同:
模型(Model)-視圖(View)-控制(Controller)模式
(圖2)
如圖2描述,MVC模式是由下面三個重要的部分組成的:
Model層實(shí)現(xiàn)系統(tǒng)的業(yè)務(wù)邏輯,通常可以用Java Bean或是EJB來實(shí)現(xiàn)。 View層用于與用戶的交互,通常用JSP來實(shí)現(xiàn)。 Controller層是Model和View之間溝通的橋梁,它用來分派用戶的請求并且選擇恰當(dāng)?shù)囊晥D用于顯示,同時,它還將用戶的輸入映射為模型層相應(yīng)可執(zhí)行的操作。
Basic portlet和MVC模式的對應(yīng)關(guān)系
(圖3)
如圖3,在Basic portlet框架中,后臺的business bean或是EJB則封裝了應(yīng)用的業(yè)務(wù)邏輯,對應(yīng)于Model層;JSP頁面對應(yīng)于 View層,它們是用戶所看到的內(nèi)容;而控制器則是一個繼承于PortletAdapter類的Portlet類,其通過actionPerformed()方法和doView()方法來完成接收用戶請求,分派調(diào)用處理函數(shù),選擇適當(dāng)?shù)囊晥D跳轉(zhuǎn)等控制工作。
下面兩個代碼片斷給出了Portlet類中actionPerformed() 和 doView()的樣例。
代碼片斷1:actionPerformed()通常負(fù)責(zé)接收用戶事件,再分派至相應(yīng)的處理函數(shù)。
public void actionPerformed(ActionEvent event) throws PortletException {
// ActionEvent handler
String actionString = event.getActionString();
// Add action string handler here
PortletRequest request = event.getRequest();
PortletSession session = request.getPortletSession();
actionBean.setSession(session);
actionBean.setRequest(request);
try {
Class actionClass = actionBean.getClass();
Method method = actionClass.getMethod("on_" + actionString, null);
method.invoke(actionBean, null);
} catch (Exception e) {
}
}
|
代碼片斷2:doView()則通常負(fù)責(zé)根據(jù)后臺的業(yè)務(wù)邏輯選擇指定相應(yīng)的頁面視圖。
public void doView(PortletRequest request, PortletResponse response)
throws PortletException, IOException {
PortletSession session = request.getPortletSession();
String JSPFile = (String) session.getAttribute(JSPPAGENAME);
if (JSPFile == null) {
// First time enter into the portle
}
// Set actionURI
setActionURIs(request, response);
// Invoke the JSP to render
getPortletConfig().getContext().include(JSPFile, request, response);
}
|
Faces portlet中和mvc模式的對應(yīng)關(guān)系
(圖4)
如圖4,后臺同樣是通過business bean或是EJB,封裝了應(yīng)用的業(yè)務(wù)邏輯,對應(yīng)于Model層;
而View層則是采用了Faces JSP(這是一種特別的JSP,它的界面是由JSF用戶界面控件組成的)。 相對而言,這里的控制層(Controller)較為復(fù)雜,每個Faces JSP頁面都對應(yīng)著一個Action Bean,負(fù)責(zé)處理該頁面上所有的用戶輸入檢驗(yàn),事件處理,以及相關(guān)頁面之間的跳轉(zhuǎn),這些都是Basic portlet中portlet類中的actionPerformed()和doView()所要做的。 看上去好像以前那個控制器已經(jīng)被拋棄了,但事實(shí)卻不是這樣,正如圖4中Controller右側(cè)的虛線框表示的那樣,它被巧妙的隱藏了起來,開發(fā)人員不會感覺到它的存在,但它卻沒有離開,依舊在后面控制著一切。
比較了Basic portlet和Faces portlet在MVC模式的實(shí)現(xiàn)上的異同之后,該來深入了解一下Faces portlet了。
深入了解faces portlet項(xiàng)目
1. 目錄結(jié)構(gòu) 通過Faces portlet wizard新建一個Portlet項(xiàng)目,分析一下目錄結(jié)構(gòu)
(圖5)
上面這張圖是一個初始的Portlet項(xiàng)目的目錄圖。
先來看WebContent
- \WebContent
這里會存放所有的JSP頁面,但是當(dāng)頁面數(shù)量很多的時候這并不是一個好辦法。也許可以將頁面分類分別存放在一些子目錄里。
- \WebContent\META-INF
這里通常會有一個METAFEST.MF文件,存放著一些和項(xiàng)目相關(guān)的元數(shù)據(jù)。我們并不需要關(guān)心它里面有些什么。
- \WebContent\theme
這里則是用來放置Portlet應(yīng)用相關(guān)的頁面樣式等,諸如css文件,這通常是美工會關(guān)心的地方。
- \WebContent\WEB-INF
這里是整個項(xiàng)目的核心重地,存放了項(xiàng)目所有重要的配置文件。
- \WebContent\WEB-INF\lib
這里存放著和Faces Portlet開發(fā)框架密切相關(guān)的一些Jar包。
- \WebContent\WEB-INF\classes
這里則存放了編譯好的一些Java類文件。
接下來,再來看Java Resources
- \Java Resources\pagecode
這里存放著和WebContent中每個JSP頁面對應(yīng)的Java類,用來處理JSP頁面的后臺動作,而所有這些都是繼承PageCodeBase這樣一個基類。
之前我們提到過,每個Faces JSP頁面都會有個Action Bean和其相對應(yīng),它們應(yīng)該在Java Resources中,那JSP文件是如何與這個Action Bean關(guān)聯(lián)起來的呢?
答案在這里,在每個Faces JSP頁面中都有一段注釋,如下面這個例子:
<%-- jsf:codeBehind language="java" location="/JavaSource/pagecode/Master.java" --%><%-- /jsf:codeBehind --%>
通過這段注釋,這個Faces JSP頁面就和/JavaSource/pagecode/Master.java 這個Action Bean建立了對應(yīng)關(guān)系。注:編輯這段注釋,你可以改變這種對應(yīng)關(guān)系,但似乎在WSAD的編輯器中是沒有辦法直接修改的,你可以用記事本直接在磁盤上修改它。
2. PageCodeBase.java (附件1中是這個類的完整內(nèi)容) 這個類是構(gòu)建一切Action Bean的基礎(chǔ),它封裝了JSF技術(shù)的一些最基本的功能。下面就抽取了其中最重要的幾個函數(shù),分別講解它們的作用。
代碼片斷3:
public PageCodeBase() {
facesContext = FacesContext.getCurrentInstance();
requestScope =
(Map) facesContext
.getApplication()
.createValueBinding("#{requestScope}")
.getValue(facesContext);
sessionScope =
(Map) facesContext
.getApplication()
.createValueBinding("#{sessionScope}")
.getValue(facesContext);
applicationScope =
(Map) facesContext
.getApplication()
.createValueBinding("#{applicationScope}")
.getValue(facesContext);
requestParam =
(Map) facesContext
.getApplication()
.createValueBinding("#{param}")
.getValue(facesContext);
}
|
這個函數(shù)是PageCodeBase這個類的構(gòu)造器,它負(fù)責(zé)獲取當(dāng)前FacesContext的實(shí)例之后,將FacesContext之間傳遞數(shù)據(jù)的4個Map結(jié)構(gòu)中的數(shù)據(jù)提取出來,分別是#{requestScope},"#{sessionScope}","#{applicationScope}","#{param}"它們分別表示在request范圍內(nèi),在session范圍內(nèi),在application范圍內(nèi)共享的變量以及頁面之間需要傳遞的參數(shù)。
代碼片斷4:
protected void gotoPage(String pageName) {
if (pageName != null) {
UIViewRoot newView =
facesContext.getApplication().getViewHandler().createView(
facesContext,
pageName);
facesContext.setViewRoot(newView);
}
}
|
這個函數(shù)實(shí)現(xiàn)了基本的頁面跳轉(zhuǎn)的功能,通過gotoPage(String pageName)這個函數(shù),頁面可以重定向到新的頁面。
代碼片斷5:
public static UIComponent findComponent(UIComponent base, String id) {
// Is the "base" component itself the match we are looking for?
if (id.equals(base.getId())) {
return base;
}
// Search through our facets and children
UIComponent kid = null;
UIComponent result = null;
Iterator kids = base.getFacetsAndChildren();
while (kids.hasNext() && (result == null)) {
kid = (UIComponent) kids.next();
if (id.equals(kid.getId())) {
result = kid;
break;
}
result = findComponent(kid, id);
if (result != null) {
break;
}
}
return result;
}
|
這個函數(shù)一般情況下不會調(diào)用,在JSF中,每個頁面中包含的所有的控件根據(jù)相互之間的嵌套關(guān)系組成一個控件樹,這個函數(shù)的作用就是在控件樹上查找并返回指定的控件實(shí)例。
代碼片斷6:
protected void putTreeAttribute(String key, Object value) {
getFacesContext().getViewRoot().getAttributes().put(key, value);
}
protected Object getTreeAttribute(String key) {
return getFacesContext().getViewRoot().getAttributes().get(key);
}
|
頁面之間很多數(shù)據(jù)都是以該頁面所對應(yīng)的控件樹的某些屬性值的形式來存儲的, 這兩個函數(shù)可以讓開發(fā)人員在代碼中存取一些需要特殊維護(hù)的數(shù)據(jù)。
代碼片斷7:
protected Object resolveExpression(String expression) {
Object value = null;
if ((expression.indexOf("#{") != -1)
&& (expression.indexOf("#{") < expression.indexOf(‘}‘))) {
value =
getFacesContext().getApplication().createValueBinding(
expression).getValue(
getFacesContext());
} else {
value = expression;
}
return value;
}
|
為什么在Faces JSP頁面中出現(xiàn)的所有變量都是以#{beanname.fieldname}來存在呢,奧秘就在這個函數(shù),F(xiàn)aces Portlet框架內(nèi)部解析變量時就默認(rèn)采用了這樣一種格式。
代碼片斷8:
protected void resolveParams(
Map paramMap,
String[] argNames,
String[] argValues,
String cacheMapKey) {
Object rawCache = getTreeAttribute(cacheMapKey);
Map cache = Collections.EMPTY_MAP;
if (rawCache instanceof Map) {
cache = (Map) rawCache;
}
for (int i = 0; i < argNames.length; i++) {
Object result = resolveExpression(argValues[i]);
if (result == null) {
result = cache.get(argNames[i]);
}
paramMap.put(argNames[i], result);
}
putTreeAttribute(cacheMapKey, paramMap);
}
|
這個函數(shù)是用來解析頁面之間傳遞的參數(shù)的,通常不需要開發(fā)人員來調(diào)用。
代碼片斷9:
protected static String getRealPath(String relPath) {
String path = relPath;
try {
URL url =
FacesContext
.getCurrentInstance()
.getExternalContext()
.getResource(
relPath);
if (url != null) {
path = url.getPath();
}
} catch (MalformedURLException e) {
e.printStackTrace();
}
return path;
}
|
這個函數(shù)用來返回當(dāng)前的FacesContext所對應(yīng)Faces JSP頁面的絕對路徑,在某些場合下非常有用。
代碼片斷10:
protected void logException(Throwable throwable) {
StringWriter stringWriter = new StringWriter();
PrintWriter printWriter = new PrintWriter(stringWriter);
throwable.printStackTrace(printWriter);
log(stringWriter.toString());
}
|
這個函數(shù)在調(diào)試程序時非常有用,它將發(fā)生的異常的StackTrace重定向到日志文件中而不是簡單的打印在控制臺上。
3. faces-config.xml 這個配置文件存儲了幾乎所有和Faces portlet密切相關(guān)的配置信息。從 http://java./dtd/web-facesconfig_1_0.dtd你可以找到它的DTD的具體內(nèi)容。
不可能解釋所有的屬性和標(biāo)簽,這里通過一個具體的例子(附件2中是它的完整內(nèi)容)來熟悉最常用的一些屬性。
代碼片斷11:
<lifecycle>
<phase-listener>com.ibm.faces.webapp.ValueResourcePhaseListener
</phase-listener>
</lifecycle>
|
這里定義了在JSF的頁面生命周期每個階段狀態(tài)的監(jiān)聽器,默認(rèn)是不需要修改的,這里使用的是IBM提供的實(shí)現(xiàn):com.ibm.faces.webapp.ValueResourcePhaseListener
代碼片斷12:
<factory>
<faces-context-factory>
com.ibm.faces.context.WPPortletFacesContextFactoryImpl
</faces-context-factory>
</factory>
|
這里則定義了和FacesContext相關(guān)的類工廠實(shí)現(xiàn),這里默認(rèn)也是IBM提供的實(shí)現(xiàn):com.ibm.faces.context.WPPortletFacesContextFactoryImpl
代碼片斷13:
<application>
<locale-config>
<default-locale>zh_CN</default-locale>
<supported-locale>en_US</supported-locale>
<supported-locale>de_DE</supported-locale>
<supported-locale>fr_FR</supported-locale>
</locale-config>
<message-bundle>messagebundle</message-bundle>
</application>
|
這里定義了這個Portlet應(yīng)用支持的locale的種類,特別指出了默認(rèn)支持的locale,并且還指定了應(yīng)用中存儲各種類型的message所需要的message bundle的名稱。
代碼片斷14:
<managed-bean>
<managed-bean-name>pc_PTestView</managed-bean-name>
<managed-bean-class>pagecode.PTestView</managed-bean-class>
<managed-bean-scope>request</managed-bean-scope>
</managed-bean>
<managed-bean>
<managed-bean-name>businessbean</managed-bean-name>
<managed-bean-class>pagecode.businessbean</managed-bean-class>
<managed-bean-scope>session</managed-bean-scope>
</managed-bean>
|
在Faces portlet應(yīng)用中,任何使用的Java Bean都會在配置文件中注冊為一個managed bean,包括變量引用時使用的名稱,具體的實(shí)現(xiàn)類名,以及它們的使用范圍(request, session, application)
代碼片斷15:
<navigation-rule>
<display-name>navigation rule sample</display-name>
<from-view-id>/master.jsp</from-view-id>
<navigation-case>
<from-action>#{pc_Master.doDetailsBtnAction}</from-action>
<from-outcome>success</from-outcome>
<to-view-id>/details.jsp</to-view-id>
</navigation-case>
<navigation-case>
<from-action>#{pc_Master.doDetailsBtnAction}</from-action>
<from-outcome>failure</from-outcome>
<to-view-id>/homepage.jsp</to-view-id>
</navigation-case>
</navigation-rule>
|
這段也是常見的配置信息,它定義了portlet應(yīng)用中各個頁面之間的跳轉(zhuǎn)關(guān)系。 from-view-id 定義了這條跳轉(zhuǎn)規(guī)則中的源頁面 <navigation-case></navigation-case> 則通過這個標(biāo)簽定義了針對一個源頁面的各種跳轉(zhuǎn)情況。 <from-action> Faces portlet中的跳轉(zhuǎn)規(guī)則是根據(jù)某個事件處理函數(shù)的返回值來判斷的,這里就定義了具體事件處理函數(shù)的名稱。 <from-outcome> 這里則定義了這種跳轉(zhuǎn)情形所對應(yīng)的事件處理函數(shù)的返回值。<to-view-id> 這里定義了在事件處理函數(shù)返回相應(yīng)的返回值后跳轉(zhuǎn)的目標(biāo)頁面。
小結(jié): Faces Portlet是一個不錯的Portlet應(yīng)用開發(fā)框架,值得去體驗(yàn)一下。如果你過去有過開發(fā)portlet應(yīng)用的經(jīng)驗(yàn),這些寶貴的經(jīng)驗(yàn)會讓你事半功倍。 希望這篇文章能夠讓你對Faces portlet印象深刻。
參考資料
- http://publib.boulder.ibm.com/infocenter/wsphelp/index.jspInformation Center of WebSphere Studio and WebSphere Application Server
- http://publib.boulder.ibm.com/pvc/wp/502/ent/en/InfoCenter/index.htmlInformation Center of WebSphere Portal Server v5.0.2
- WSAD V5.1.2的聯(lián)機(jī)幫助,它會隨WSAD一起安裝。
- http://www.ibm.com/developerworks/cn/cnwsdd.nsf/wsdd-onlinecourse-bynewest/7DF1D2C22B4ACCD9C8256E7E00230C9D?OpenDocument教程:創(chuàng)建基于 Web 的用戶界面--使用 WebSphere Studio V5.1.1 來開發(fā) JavaServer Faces 應(yīng)用程序
- http://www.ibm.com/developerworks/cn/websphere/library/techarticles/0403_lynn/0403_lynn.htmlHello World -- WebSphere Portal V5 最簡單的 portlet
- http://www./en/jsr/detail?id=127JSR127,這是JCP通過的JSF規(guī)范
- http://www./非常不錯的一個關(guān)于JSF的開發(fā)社區(qū)
- http://www.ibm.com/developerworks/cn/websphere/zones/studio/theme/studionewtech.htmlWSStudio新功能入門專題
- 在 www.google.com上搜索關(guān)鍵字MVC,你可以很快找到很多關(guān)于MVC的材料。
|