<rt id="bn8ez"></rt>
<label id="bn8ez"></label>

  • <span id="bn8ez"></span>

    <label id="bn8ez"><meter id="bn8ez"></meter></label>

    Sealyu

    --- 博客已遷移至: http://www.sealyu.com/blog

      BlogJava :: 首頁 :: 新隨筆 :: 聯(lián)系 :: 聚合  :: 管理 ::
      618 隨筆 :: 87 文章 :: 225 評論 :: 0 Trackbacks

    Google Web工具包(GWT)確實是使用Java開發(fā)Ajax應(yīng)用的一種誘人方法。如果你在AWT/Swing/SWT和服務(wù)器小程序方面有著扎實背景,實際上很容易學會使用GWT,但如果要做的不僅僅是快速原型設(shè)計,那么某些難題仍然存在。

    忠告之一: 分而治之

    眾所周知,GWT應(yīng)用就是Java應(yīng)用。不過,問題在于是“哪種Java”,我們需要牢記: GWT編譯的是與J2SE 1.4.2或者更早版本兼容的Java源代碼。另外,只有J2SE 1.4.2 API的子集得到支持,即java.lang和java.util程序包。即便在使用這些程序包時,也要非常認真地研究Google在運行庫支持方面的注 釋,并且牢記相應(yīng)的忠告: 如果確保從一開始就只使用客戶端代碼中的可轉(zhuǎn)換類,那么就可以避免許多問題。為了及早發(fā)現(xiàn)問題,只要在宿主模式(hosted mode)下運行,就要對照JRE仿真庫檢驗代碼。因而,第一次運行應(yīng)用時,就會發(fā)現(xiàn)大部分不支持的庫。所以,要及早并且經(jīng)常運行。

    現(xiàn)在,筆者給出的忠告就是“分而治之”,具體意思就是一開始就把應(yīng)用代碼分成三個不同的部分: 客戶端代碼、RPC相關(guān)代碼和服務(wù)器端代碼,然后構(gòu)建相應(yīng)的Eclipse項目,從而完成任務(wù)。這樣一來,就可以利用不同的Java語言版本,用于客戶端 和服務(wù)器部分。筆者用Java 5構(gòu)建了應(yīng)用的服務(wù)器部分(服務(wù)器小程序代碼); 但如果使用Mustang版本,那么在本文的代碼片段中(由于篇幅有限,本文所涉及的程序代碼可通過以下鏈接查詢: http://blog.ccw.com.cn/article-htm-itemid-17924-type-blog.html),可以用Java 6取代Java 5。即便在服務(wù)器端仍然使用J2SE 1.4.2,這種分治法也可以在將來提供更大的靈活性,明確分離代碼(“分離問題”),而不會在GWT宿主模式下限制調(diào)試操作。如果所有部分都在一個 Eclipse項目中,則需要非常嚴謹,特別是在服務(wù)器端上; 不然,就會出現(xiàn)編譯或者運行問題。

    需要使用特殊的命名約定,這樣可以清楚確認不同項目,并且簡化部署腳本??梢允褂闷┤缑麨镚WT-< ModuleName>的Eclipse工作集來包括所有三個項目。這里,“ModuleName”是識別Web應(yīng)用的GWT模塊的名稱。

    ● 客戶端代碼: 包含與用戶界面相關(guān)的代碼,可以轉(zhuǎn)換成JavaScript。因此,局限于J2SE 1.4.2和GWT運行時支持。啟用每個項目的Eclipse Java編譯器設(shè)置和“Java編譯器錯誤/警告”,把Java依從級別調(diào)整到1.4、把源代碼和類文件兼容性調(diào)整到1.4(假設(shè)不是使用1.4之前的 JDK版本)。該項目的名稱是< ModuleName>-client,譬如“JUnit2MR-client”,它依賴于構(gòu)建路徑設(shè)置中的< ModuleName>-rpc項目。程序包名稱類似< com.company.project>.gwt.< moduleName>.client。

    ● RPC相關(guān)代碼: 包含RPC相關(guān)的代碼,可以轉(zhuǎn)換成JavaScript。該項目遵從與上述客戶端代碼項目同樣的指導準則。項目名是< ModuleName>-rpc,譬如“JUnit2MR-rpc”,它并不依賴于其他任何項目。程序包名稱與< ModuleName>-client項目的程序包名稱一樣。RPC項目包含客戶端上的遠程接口、RPC期間由GWT進行序列化的數(shù)據(jù)傳輸對象,以 及全局常量類。

    ● 服務(wù)器端代碼: 含有服務(wù)器小程序代碼,如果服務(wù)器端由Java服務(wù)器小程序組成的話。如果使用Tomcat 5.5或者Tomcat 6,可以充分利用Java 5+的全部功能。啟用每個項目的Eclipse編譯器設(shè)置,然后使用Java 5編譯器設(shè)置,依從級別設(shè)置為5.0。如果使用Eclipse 3.2.2,那么其新的“源代碼→清理”特性也值得配置。該項目名稱是< ModuleName>-server,譬如“JUnit2MR-server”,它依賴于構(gòu)建路徑設(shè)置中的< ModuleName>-rpc項目。如果按照GWT的默認程序包提案進行編程,程序包名稱是< com.company.project>.gwt.< moduleName>.server。

    忠告之二: 調(diào)試和錯誤報告不僅僅只有Window.alert ()

    在創(chuàng)建GWT應(yīng)用時,其實可以使用IDE的全部調(diào)試功能。但在深入分析何處可能出現(xiàn)錯誤之前,需要代碼的客戶端和 服務(wù)器端都有可靠的異常報告機制。使用try/catch代碼塊通??梢宰龅竭@一點。在客戶端的catch代碼塊中,應(yīng)當注意這一現(xiàn)實: 默認的方法調(diào)用e.printStackTrace()并不是在所有情況下都適合的解決辦法。它適用于應(yīng)用運行在GWT宿主模式下,把文本輸出到 Eclipse控制臺。不過在Web模式下,要問問自己: “我發(fā)送到stdout或者stderr的堆棧跟蹤信息和錯誤信息會在什么地方顯示?”一種可能的解決方法就是使用Mat Gessel的調(diào)試實用程序類(http://www.asquare.net/gwttk),但是需要瀏覽器JavaScript控制臺來查看Web模 式下的結(jié)果。

    在客戶端,建議要做的一件事就是,使用GWT.setUncaughtExceptionHandler()方 法,為任何未被發(fā)現(xiàn)的異常提供自己的異常處理程序。發(fā)現(xiàn)了這幾種異常后,有幾個選擇: GWT.log(message, caught)、Debug.println (message_with_stacktrace); 如果使用Mat Gessel的Debug類,可選擇Window.alert(message_with_stacktrace),或者自己定制的錯誤報告。

    視來源而定,會得到“無法裝入模塊”或者“未被發(fā)現(xiàn)的異常被漏過”的信息。筆者編寫了一個小小的DebugUtility類,它提供了易于定制的默認客戶端錯誤處理機制(見代碼片段1)。

    在服務(wù)器端,可以使用java.util.logging API或者log4j的廣泛功能,具體取決于個人偏好或者項目的約束條件。但要是沒有為GWT的 com.google.gwt.user.server.rpc.RemoteServiceServlet類打補丁,對于未被發(fā)現(xiàn)、未被檢查的異常,只 會在堆棧跟蹤里面得到提示,指向生成該錯誤的服務(wù)器端類。對于catch()代碼塊里面發(fā)現(xiàn)及報告的被檢查的異常,一切都正常。

    忠告之三: 當心GWT Shell的“刷新”按鈕陷阱

    在宿主模式下啟動應(yīng)用時,會在瀏覽器任務(wù)欄上看到“刷新”按鈕。要是摁了這個按鈕,GWT就會把修改過的Java 客戶端源代碼重新編譯成Java字節(jié)碼(作為.gwt.-cache/bytecode目錄中的.tmp文件),然后重新裝入模塊??梢允褂眠@個按鈕來縮 短編輯→編譯→調(diào)試周期,但在使用這項特性時要牢記幾個方面:

    ● 只有修改過的源代碼才重新編譯,也就是說,不會為依賴修改過代碼的文件生成新的字節(jié)碼。所以,如果改變了全局常量的值,假設(shè)public final int字段的值,不會立即在相關(guān)文件看到這個變化。

    ● 只有修改過的源代碼才由GWT重新編譯。這意味著,即便Eclipse IDE里面的“Project clean”也幫不上忙; 要影響到所有的相關(guān)源代碼,譬如通過添加新的空行。

    因為這個過程相當笨拙,筆者的忠告是在修改全局常量時遵循以下四個步驟:

    1.在相應(yīng)的源文件里面改變public final constant值;

    2.重新編譯改變后的源代碼;

    3.移除整個< ModuleName>-client/.get-cache/bytecode目錄,從而刪除GWT緩存內(nèi)容;

    4、使用Eclipse里面的“Run as”,重新開始啟動應(yīng)用,從而創(chuàng)建帶重新編譯后字符碼的新GWT緩存內(nèi)容,這種情況下,最好忽視“刷新”按鈕,不過在有些情況下,刪除整個< ModuleName>-client/.get-cache/bytecode目錄后可以使用“刷新”按鈕。

    在修改服務(wù)器端代碼時,GWT字節(jié)碼緩存內(nèi)容不受影響。不過,嵌入的Tomcat實例會緩存它,因而在使用“刷新”按鈕后,只有重新開始啟動應(yīng)用后最初改變的代碼才會得到認可。所以為了安全起見,改變服務(wù)器端代碼后,最好還是重新開始啟動應(yīng)用。

    忠告之四: 在宿主模式下讀取Servlet Init參數(shù)

    在處理數(shù)據(jù)庫系統(tǒng)時,一般不希望服務(wù)器小程序源代碼中有硬編碼的數(shù)據(jù)庫連接參數(shù)。通常會從屬性文件讀取這些參數(shù); 或者更好的是,把它們作為init參數(shù)提供給服務(wù)器小程序(作為應(yīng)用的Web.xml文件的一部分)。如果在Web模式下運行應(yīng)用那沒有什么,但在宿主模 式下會出問題,這是由于GWT宿主模式下的服務(wù)器小程序處理存在限制。

    好消息是,只要修改由嵌入式Tomcat實例使用的Web.xml文件,就可以解決這個問題。為此,修 改< ModuleName>-client/tomcat/webapps/ROOT/WEB-INF目錄中的Web.xml文件(或者必要時創(chuàng)建一 個): 除了嵌入式Tomcat的GWTShellServlet映射外,添加帶有init參數(shù)的上下文部分。因為上下文信息是“全局性的”,而不是針對特定的服 務(wù)器小程序,在這里只有一部分的init參數(shù)信息,或者使用特殊的命名方案,把參數(shù)與不同的服務(wù)器小程序聯(lián)系起來。如果使用這個新的web.xml文件, 可以刪除src/web/WEB-INF文件夾中的那個舊文件。

    在服務(wù)器小程序代碼中,訪問init參數(shù)的方式與Web模式下讀取它們的方式一樣,譬如final String host = getInitParameter("host")。筆者實現(xiàn)這一點的辦法就是修改GWT的RemoteServiceServlet,方法跟第二個忠告 里面的如出一轍。現(xiàn)在,只要覆蓋GenericServlet的getInitParameter()方法,以便使用 getServletContext(),而不是 getServletConfig()。

    另一個忠告是,如果在宿主模式下和Web模式下測試不同的服務(wù)器代碼,略過Gant腳本中的GWT編譯部分,從“temp”位置拷貝編譯前的JavaScript代碼,則可以節(jié)省時間。這適用于客戶端代碼復雜、編譯時間超過10分鐘的情形。

    忠告之五: 在瀏覽器里面顯示PDF文件

    大多數(shù)實際的Web應(yīng)用提供了生成及閱讀PDF文件的方法。本文假設(shè)這個PDF文件由服務(wù)器小程序生成,譬如通過 JasperReport。以后只要點擊某個超文本鏈接,就可以在瀏覽器里面閱讀生成的文件。如果想在宿主模式下和Web模式下測試這項特性,建議采取以 下步驟:

    1.設(shè)計一個RPC接口,接受告訴服務(wù)器是在宿主模式下運行還是在Web模式下運行的布爾參數(shù)。接口方法會返回的字符串應(yīng)當帶有服務(wù)器小程序生成的PDF文件的名稱(即文件名的最后一部分)。

    2.根據(jù)代碼片段4顯示的代碼,實現(xiàn)服務(wù)器小程序代碼,這取決于布爾參數(shù)“isScript”。

    3.在客戶端: 在窗口組件代碼里面,使用GWT.isScript()參數(shù)調(diào)用createXyzPDF()方法,從而生成包含服務(wù)器小程序結(jié)果字符串的外部超文本鏈接。

    代碼片段4顯示了接口方法名為createSummaryPDF()的示例。從服務(wù)器小程序返回的字符串是“summary.pdf”。

    這當然不是處理這種情況的惟一辦法,但目前適用于我們這個示例。請注意: 在宿主模式下啟動應(yīng)用之前,必須在< ModuleName>-client project's src/…/public文件夾中至少創(chuàng)建一個虛假的“summary.pdf”文件(文件名從服務(wù)器小程序返回)。不然,在瀏覽器中點擊了超文本鏈接 后,GWT試圖讀取PDF文件時,會出現(xiàn)“HTTP 404-找不到網(wǎng)頁”的信息。

    忠告之六:力求獲得無狀態(tài)服務(wù)器

    設(shè)計客戶機/服務(wù)器Web應(yīng)用時要考慮的一個關(guān)鍵問題就是: 如何處理會話和狀態(tài)管理?在Web 1.0時代,答案很顯然: 會話和狀態(tài)管理是一個服務(wù)器問題。但若使用GWT,就有另一個選擇。服務(wù)器再也不是只提供HTML內(nèi)容的“web”服務(wù)。使用GWT RPC,服務(wù)器現(xiàn)在可以支持只提供結(jié)構(gòu)化數(shù)據(jù)的服務(wù)———在本文示例中,服務(wù)由服務(wù)器小程序?qū)崿F(xiàn)。

    那么,GWT對會話和狀態(tài)管理有何影響呢?GWT的技術(shù)領(lǐng)導Bruce Johnson在去年的JAOO大會上指出,若使用GWT,會話管理現(xiàn)在應(yīng)當是一個客戶端問題。附圖顯示的幻燈片評述了種種變化。

    在本文的JUnit2MR GWT應(yīng)用中,筆者一開始使用傳統(tǒng)方法來處理服務(wù)器小程序中的會話狀態(tài)。但這是相當笨拙的任務(wù),于是尋找另一種選擇。因此,看了Bruce的幻燈片后,決 定重新設(shè)計整個應(yīng)用。但這一步需要改變所有RPC接口、緩存策略; 最重要的是,還要改變所有的服務(wù)器小程序。因此筆者的建議是: 及早考慮在何處實施會話和狀態(tài)管理,不妨試試Bruce Johnson的訣竅。最終會收到成效。

    由于這個決定,客戶端對象之間有了更多的聯(lián)系。于是筆者使用了有名的GoF中介者模式(mediator pattern)。不過,在客戶端有一些JDK 1.4和GWT運行庫的限制。因此,重新實現(xiàn)了PropertyChangeEvent類和中介者支持,來處理監(jiān)聽程序注冊和消息廣播。

    忠告之七: 使用Selenium實現(xiàn)GWT Web測試的自動化

    Selenium是一種開源工具,它能夠輕松測試包含豐富、互動的客戶端內(nèi)容的Web應(yīng)用。 所以,它非常適用于測試像用GWT創(chuàng)建的應(yīng)用那樣的Ajax應(yīng)用。

    當然,GWT里面仍有JUnit和JUnit支持功能,特別是針對系統(tǒng)的異步部分。這里著重介紹 Selenium,因為它易于使用(至少它的IDE是這樣)、功能強大。最后但并非最不重要的一點是,它與JUnit有許多共同之處。可以使用 Selenium IDE來記錄GUI用例,然后使用其“Play”特性來運行記錄下來的操作。每個操作之后跟著類似JUnit的“assert”命令,負責確認頁面上的某 些文本。該IDE是Firefox的擴展插件,但務(wù)必要使用最新版本的Selenium: Selenium IDE 0 .8 .7,因為它包含了“waitFor…”命令的重大修正版。說到測試Ajax應(yīng)用,這些命令以及“pause”命令非常重要。

    忠告之八: 使用Groovy Gant腳本部署應(yīng)用

    在GWT宿主模式下試運行應(yīng)用,這確實很好,但把應(yīng)用部署到應(yīng)用服務(wù)器上或者類似Tomcat的服務(wù)器小程序容器 上,GWT的真實功能才會體現(xiàn)出來。在這一步,需要創(chuàng)建一個war文件,它會自動拷貝到Tomcat“webapps”目錄。當然,可以使用Ant和 ant-contrib進行所有必要的準備、編譯、拷貝及其他任務(wù)。但由于Ant腳本變得更復雜后, ant-contrib控制結(jié)構(gòu)和屬性regex處理有一點笨拙。于是可以使用集Groovy和Ant兩者之所長的Gant。安裝Groovy和Gant 用不了10分鐘,然后,使用來自“build.properties”文件的普通屬性,即可定制“build.gant”腳本。(小黑編譯)

    (計算機世界報 2007年6月25日 第24期 B22、B23)

    posted on 2010-01-28 17:47 seal 閱讀(1176) 評論(0)  編輯  收藏 所屬分類: GWT
    主站蜘蛛池模板: 91av视频免费在线观看| 手机在线毛片免费播放| 亚洲av专区无码观看精品天堂| 国产福利在线观看免费第一福利| 亚洲av无码av在线播放| 亚洲精品自在在线观看| 国产精彩免费视频| 一级毛片大全免费播放下载| 亚洲综合久久成人69| 亚洲国产电影av在线网址| 99久久99久久免费精品小说| 曰批全过程免费视频免费看| 91久久亚洲国产成人精品性色| 国产成人aaa在线视频免费观看| 亚洲视频在线免费观看| 偷自拍亚洲视频在线观看| 久久亚洲精品成人av无码网站| 免费国产在线观看不卡| 国产精彩免费视频| 野花香高清视频在线观看免费| 亚洲av最新在线观看网址| 亚洲国产精品不卡在线电影| 国产成人亚洲综合无码| 在线免费不卡视频| 人人揉揉香蕉大免费不卡| 免费大片av手机看片高清| 国内精品久久久久影院亚洲| 亚洲av无码国产精品夜色午夜| 又黄又爽一线毛片免费观看 | 亚洲欧美一区二区三区日产| 亚洲AV日韩AV永久无码下载| 亚洲高清无码专区视频| 免费观看成人毛片a片2008| 91精品国产免费入口| a级毛片免费播放| 成人久久久观看免费毛片| 亚洲一区二区观看播放| 亚洲国产成人久久99精品| 西西人体44rt高清亚洲 | 亚洲av永久无码精品天堂久久| 亚洲精品无码不卡在线播放HE|