Java很早就推出了Java Web Start(簡稱JWS)技術。這一技術的初衷很好:但願將桌面程序和Web頁面之間搭起一個無縫的橋樑。雖然Applet技術已經存在了十多年,可是它 日趨老邁衰落,因此JWS也就應運而生了。 可是JWS並未順利實現它的初衷。從Java的幾回大改版均可以看到,JWS的bug多多,漏洞頻頻,Sun和Oracle不得不頻繁的進行打補丁修復。 能夠看看Java 5和6每次大小版本升級變化中,有多少是和Java Web Start有關的。難怪不少人都這樣感嘆:「哥不再用Java Web Start部署應用了!」其實也未必,隨着Java的不斷完善,咱們只要瞭解更多的技巧,就能夠有效的消除一些JWS潛在的問題,並順利的應用在企業應用 中。 以2BizBox ERP項目爲例,本文介紹如何在企業應用中利用動態生成JNLP文件的技術來實現應用的快速部署。html
你們知道,2BizBox ERP做爲一個免費的高質量ERP軟件,有成千上萬的用戶。就咱們開發團隊負責維護的服務器,就有近千臺。每臺服務器都是一家企業,每家企業又有幾十上百 的客戶端。若是採用下載客戶端安裝程序進行安裝的方式來維護諸多的客戶端,無疑是巨大的工做量,用戶和咱們開發團隊都不會輕鬆方便。爲了解決這一問題,採 用JWS無疑是必然的選擇。 java
爲了讓客戶端自動啓動下載和安裝程序,咱們在企業的2BizBox ERP服務器上部署如下JNLP文件內容:web
<?xml version="1.0" encoding="utf-8"?> <jnlp spec="1.0+" codebase="http://**.**.**.**/webstart/"> <information> <title>2BizBox</title> <vendor>Serva Software</vendor> <homepage href="http://www.2bizbox.com" /> <description>2BizBox ERP 3</description> <offline-allowed /> </information> <security> <all-permissions /> </security> <update check="always" policy="always" /> <resources> <j2se href="http://java.sun.com/products/autodl/j2se" version="1.6+" initial-heap-size="128m" max-heap-size="512m" /> <jar href="2bizbox.jar" /> <jar href=" lib1.jar" /> <jar href="lib2.jar" /> <jar href=" lib3.jar" /> <jar href="lib4.jar" /> <!-- more jar. --> </resources> <application-desc main-class=" com.serva.bb2.gui.Main "> <argument>**.**.**.**</argument> </application-desc> </jnlp>
上面的JNLP文件定義了2BizBox ERP客戶端啓動所須要的jar包以及下載位置、jre版本等。後端
在實際應用中,效果良好。可是因爲JNLP和JWS自己的bug,在某些狀況下,後臺jar程序更新升級後,用戶側啓動jnlp並不能得到更新,須要強行 清空JWS緩存才行,這確定不是通常用戶懂得的。還有一種狀況,就是因爲ERP自己的jar包發生了變化(例如發生了增減),此時至關於jnlp文件的內 容發生了變化。這時候,要求用戶一側機器必須意識到jnlp的變化並先將jnlp進行更新。在不少java版本中(例如jre6的早期版本——例如 jre6 update20以前),因爲潛在的一些bug等緣由,都不能順利的進行更新,致使程序啓動失敗。瀏覽器
如何解決這一狀況呢?採用動態jnlp是一個有效的方法。緩存
動態jnlp的思路是:在服務器的後端,經過jsp或servlet來動態的生成一個jnlp文件,而不是放置一個靜態的固定不變的jnlp文件。這樣,jnlp文件內容就能夠經過後臺應用的邏輯進行動態生成建立:須要什麼jar包、須要什麼jre版本等等。服務器
以jsp爲例。在這個jsp中,首先要注意的幾個技術點是:要設置本頁面不要被瀏覽器緩存,放置jnlp內容變化沒法及時被更新;其次要設置mime類型 讓瀏覽器認爲它是一個jnlp文件,以便下載執行而不是直接在瀏覽器中顯示出來。經過設置response便可達到這些目的: app
response.setHeader("Pragma", "no-cache"); response.setHeader("Expires", "0"); response.setHeader("Content-Disposition", "filename=\"bb.jnlp\";"); response.setContentType("application/x-java-jnlp-file");
其中,禁止瀏覽器和webstart緩存jnlp內容,經過設置:response.setHeader("Pragma", "no-cache");和response.setHeader("Expires", "0"); jsp
設置文件類型,並給定一個動態的文件名。這個經過這個進行:response.setHeader("Content-Disposition", "filename=\"bb.jnlp\";");response.setContentType("application/x-java-jnlp-file");函數
一個須要注意的問題是,在動態生成jnlp文件時,要注意jnlp文件中的href標籤不要進行設置。爲何呢?看一下jnlp的格式文檔是這樣說的:http://lopica.sourceforge.net/ref.html#jnlp
The jnlp file's one and only root.
Attributes
spec=version , optional
Specifies what versions of the jnlp spec a jnlp file works with. The default value is 1.0+. Thus, you can typically leave it out.
version=version , optional
Specifies the version of the application as well as the version of the jnlp file itself.
codebase=url , optional
Specifies the codebase for the application. Codebase is also used as base URL for all relative URLs in href attributes.
href=url , optional
Contains the location of the jnlp file as a URL. If you leave out the href attribute, Web Start will disable the update check on your JNLP file, and Web Start will not treat each new JNLP file as an application update - only updated jar files will. Leaving out href usually makes only sense if your jnlp file is created dynamically (that is, throug a cgi-script, for example) and if your jnlp file's arguments or properties change from request to request (user to user).
Note, that Java Web Start needs href to list your app in the Web Start Application Manager.
可見在動態生成jnlp時候就不要設置href了,這樣就能夠保證每次瀏覽器會從新下載jnlp文件內容,不然可能會被緩存,沒法及時更新程序。
另一個技巧是:jnlp文件中的jar包,能夠進行動態檢查文件jar包並動態生成。這樣,若是之後程序的jar文件有增減,就沒必要修改jnlp文件 了。方法也很簡單:檢查當前web在服務器的絕對路徑,並list全部的jar文件,而後在jnlp生成時候輸出便可:
<% String urlString=request.getRequestURL().toString(); URL url=new URL(urlString); String host=url.getHost(); String path = request.getSession().getServletContext().getRealPath("/"); path=path.replace("\\.\\", "\\"); File file=new File(path); String[] files = file.list(); ArrayList jarNames=new ArrayList(); for(int i=0;i<files.length;i++){ String fileName=files[i]; if(fileName.toLowerCase().endsWith(".jar")){ jarNames.add(fileName); } } %>
而後在jar的部分這樣列出:
<resources> <j2se href="http://java.sun.com/products/autodl/j2se" version="1.6+" initial-heap-size="128m" max-heap-size="512m"/> <% for(int i=0;i<jarNames.size();i++){ out.write("\n"); out.write("<jar href=\""+jarNames.get(i).toString()+"\"/>"); } %> </resources>
最後,若是須要在jnlp中指定當前服務器的ip地址或主機地址,也能夠經過動態生成。例如jnlp文件中的codebase,就是如此。另 外,2BizBox ERP還須要在主函數中給出當前服務器的ip地址。而對於上千家的2BizBox服務器,每一個jnlp要手工維護ip地址,是不可想象的。這裏經過動態生 成,就永遠的解決了這個問題:
String urlString=request.getRequestURL().toString(); URL url=new URL(urlString); String host=url.getHost();
而後在jnlp中:
<jnlp spec="1.0+" codebase="http://<%=host%>/webstart/"> <application-desc main-class="com.serva.bb2.gui.Main"> <argument><%=host%></argument> <application-desc>
這樣,經過jsp動態生成jnlp的方案就完成了。它在2BizBox ERP中應用良好,方便的讓上千家2BizBox ERP的雲主機用戶快速獲得程序更新,而簡化了程序的維護方式。