AppFuse是一個(gè)集成了當前最流行的Web應用框架的一個(gè)更高層次的Web開(kāi)發(fā)框架,也可以說(shuō)是一個(gè)Web開(kāi)發(fā)基礎平臺,它與它所集成的各種框架相比,它提供了一部分所有Web系統開(kāi)發(fā)過(guò)程中都需要開(kāi)發(fā)的一些功能,如登陸、用戶(hù)密碼加密,用戶(hù)管理、根據不同的用戶(hù)可以展現不同的菜單,可以自動(dòng)生成40%-60%左右的代碼,自帶了默認的一些在CSS中設定的樣式,使用這些樣式能很快的改變整個(gè)系統的外觀(guān),還有自動(dòng)化測試的功能。
它最大的價(jià)值就是為我們提供了一個(gè)Web開(kāi)發(fā)的新的方式和思路,盡管這些技術(shù)在國外都已進(jìn)很流行了,但在國內能夠將Hibernate、Struts、Spring、DBUnit、Ant、Log4J、Struts Menu、Xdoclet、SiteMesh、Velocity、JUnit、JSTL、WebWork這些技術(shù)集成到一個(gè)框架中的還不多見(jiàn),所以即使不使用它的全部功能,它也給我們提供了一個(gè)很好的借鑒、學(xué)習的機會(huì )。
通過(guò)關(guān)注AppFuse,我們可以看到目前國外的主流開(kāi)發(fā)都使用了哪些技術(shù),開(kāi)發(fā)方式是什么樣的,可能達到什么樣的結果,而在以前,是很少能夠看到這樣完整的例子的。
AppFuse的另一個(gè)啟示是:我們可以依靠開(kāi)源軟件的功能降低開(kāi)發(fā)成本,而且可以閱讀開(kāi)源軟件的代碼提高所在團隊的整體實(shí)力。
AppFuse的作者 matt raible是當今開(kāi)源世界一個(gè)比較活躍的開(kāi)發(fā)者,它是AppFuse、Struts Menu的作者,也是XDoclet、DisplayTag等一些著(zhù)名開(kāi)源項目的積極參與者,《Hibernate In Action》的作者就在感謝的名單里面提到他,XDoclet的下載版本中所帶的Hibernate標簽部分的例子就是他寫(xiě)的,他還是2004年Apache技術(shù)年會(huì )的主講人之一。(這些都是我這2個(gè)多月來(lái)搜集到的,呵呵)
但是通過(guò)2個(gè)月的實(shí)際學(xué)習和使用,我也遇到一系列的問(wèn)題,因為AppFuse是將其他的一些類(lèi)庫或者框架集成在一起的,集成的技術(shù)眾多,而且有一些技術(shù)在國內甚至很少有人知道,資料也比較少,所以雖然作者經(jīng)過(guò)了一些測試,但都是基于英文編碼的,而對于中文編碼來(lái)說(shuō),還潛在的存在著(zhù)一些問(wèn)題,雖然不是AppFuse的問(wèn)題,但卻降低了開(kāi)發(fā)速度,下面是我在開(kāi)發(fā)過(guò)程中遇到過(guò)的問(wèn)題,有些解決了,有些還沒(méi)有解決:
一.Struts
1. AppFuse中默認的MVC框架是Struts,而且使用的是LookupDispatchAction,并且使用的是按鈕(button),在XP下用IE瀏覽效果還可以,但如果在2000或者98下,就使外觀(guān)很難看,而且當時(shí)我還遇到一個(gè)問(wèn)題:如果按鈕顯示中文,則在DisplayTag中翻頁(yè)失靈,而且報錯,后來(lái)我把BaseAction的相關(guān)方法改變了,才可以使用,因為國內的客戶(hù)都比較重視界面,所以后來(lái)我將那些按鈕都改成圖片了,當然也要添加一些方法了,有點(diǎn)麻煩!
2. Struts中的標簽如今推薦使用的只有html部分的標簽了,其他的標簽或者可以使用JSTL替代,或者已經(jīng)不推薦使用了,而且AppFuse中推薦使用JSTL,而JSTL和struts的標簽的聯(lián)合使用時(shí),需要的不是<標簽>,而是<標簽>,這個(gè)問(wèn)題曾經(jīng)困擾了我整整2天。
3. Struts的Validation的校驗規則并不完善,比如如果使用客戶(hù)端的javascript校驗,則在郵箱中輸入漢字根本校驗不出來(lái),到了服務(wù)器端報錯。
4. 最嚴重的問(wèn)題是AppFuse生成的Struts的validation.xml文件中有許多多余的“.”,如果你去掉了,常常在執行ant的deploy任務(wù)時(shí)又恢復原樣。這樣是提交表單的時(shí)候經(jīng)常會(huì )報javascript的腳本錯誤或者缺少對象或者缺少value,所以我會(huì )手工的修改這個(gè)文件,然后把修改后的文件備份,當重新生成有錯誤的文件時(shí),我會(huì )用備份的沒(méi)有錯誤的文件去覆蓋。
5. Struts的validatioin對于使用同一個(gè)FormBean的Action的校驗方式比較復雜。(待解決)。
二.Hibernate
1. Hibernate是現在受到越來(lái)越多的人推崇的一個(gè)ORM工具(框架、類(lèi)庫),它將我們從繁瑣的使用JDBC的開(kāi)發(fā)過(guò)程中解放出來(lái),但同時(shí)也帶來(lái)了新的問(wèn)題,如學(xué)習曲線(xiàn),執行效率,數據庫設計優(yōu)化,還有最重要的靈活性。Hibernate不是一個(gè)很容易上手的東西,要完全駕馭它還需要讀很多資料,但好的資料卻很少。
2. 使用Xdoclet可以很方便的生成Hibernate中的持久類(lèi)的配置文件(*.hbm.xml),但對一些特殊的映射卻無(wú)能為力,如使用序列的id生成規則,序列的名字沒(méi)有地方寫(xiě),所以也只好先利用它生成主要的內容,然后手工修改。
3. 同樣還是id的生成策略問(wèn)題,如果使用序列、hilo等需要一些數據庫機制支持的策略時(shí),schemaExport并不能自動(dòng)生成序列或者保存當前id的表,這項工作仍然要手工解決。
4. Hibernate中提供了幾種關(guān)聯(lián),一對一、一對多、多對多,但對于怎樣調整效率卻沒(méi)有一個(gè)很明確的提示,還要根據情況判定,這就帶來(lái)和一些彈性的設計。
5. Hibernate中可以選擇的操作數據庫的方式有3種,其中HQL功能最強大,但有些功能使用標準查詢(xún)可能會(huì )更方便,但會(huì )有一些限制,所以雖然它很靈活,但易用性不如JDBC好。
三.Spring
在A(yíng)ppFuse的過(guò)程中,Spring完全隱藏在幕后,除了一些配置外,幾乎感覺(jué)不到它的存在,所以我在使用它的過(guò)程中并沒(méi)有遇到什么麻煩,這里只是簡(jiǎn)單的介紹一下它在A(yíng)ppFuse中起到的作用。
1. Spring在A(yíng)ppFuse中起到的主要作用是對Hibernate的Session和事務(wù)的管理,利用Spring封裝的Hibernate模板類(lèi),我們大大地減少了實(shí)現DAO的代碼行數。
2. Spring還起到了連接映射文件和類(lèi)之間的關(guān)聯(lián),及接口和實(shí)現類(lèi)之間的關(guān)聯(lián),這些都依賴(lài)于Spring的IoC的機制的實(shí)現。
3. 對于字符進(jìn)行編碼和解碼部分用到了Spring自帶的Filter,只需要在配置文件中配置就好了。
四.SiteMesh
SiteMesh是一個(gè)基于Decorator模式的技術(shù),它可以修飾返回的網(wǎng)頁(yè)文件,它的工作方式受到越來(lái)越多的人的推崇,這點(diǎn)從Manning出版的一些技術(shù)書(shū)籍中可以看出來(lái)。
我在使用SiteMesh的過(guò)程中并不順利,我參考了《Java Open Source Programming》,這本書(shū)中說(shuō)SiteMesh在默認的情況下不對下載文件進(jìn)行裝飾,但我在下載文件時(shí)發(fā)現,我的文件內容被丟棄了,取而代之的是SiteMesh的模板的內容,后來(lái)我通過(guò)修改SiteMesh的配置文件解決了這個(gè)問(wèn)題,但感覺(jué)還有一些不太清楚的地方需要學(xué)習。
五.DisplayTag
DisplayTag是一個(gè)優(yōu)秀的顯示內容的標簽,從SourceForge的訪(fǎng)問(wèn)量來(lái)看,它是很活躍的項目,僅次于A(yíng)nt、Hibernate、Xdoclet等幾個(gè)著(zhù)名的項目,我總結,它的主要功能有4項:顯示、分頁(yè)、排序、將顯示的數據寫(xiě)入指定類(lèi)型的文件中,然后下載。
1. 據我使用的情況看,我只使用了分頁(yè)和顯示的功能,因為當時(shí)我沒(méi)有很好的解決中文編碼的問(wèn)題,所以排序會(huì )有問(wèn)題,直到昨天,我在朋友的幫助下解決了這個(gè)問(wèn)題,至此我可以放心使用的功能又增加了排序(我昨天簡(jiǎn)單的測試了一下是可以的)。
2. 但對于將顯示的內容生成到一個(gè)指定格式的文件中的功能卻有著(zhù)很多缺陷,如:
(1) 生成的文件中只有顯示的數據,那些沒(méi)有顯示在界面上的的數據,則不會(huì )被寫(xiě)到文件中。
(2) 如果修改了DisplayTag的顯示的內容,比如添加一列,在這列中的內容不是字符,而是HTML的標簽,則生成的文件只有這些HTML標簽,而沒(méi)有數據。
(3) 即使DisplayTag中沒(méi)有我們定制的HTML腳本,生成的文件偶爾也有問(wèn)題,比如:它會(huì )把“007”生成為“7”,把字符串自動(dòng)的轉換為整型值。有時(shí)候還生成空白內容的文件。
(4) DisplayTag生成的Excel文件兼容性不好,有時(shí)在Excel2003中不能正常打開(kāi),或者在XP下打開(kāi)報錯。
后來(lái),我看了作者寫(xiě)的《Spring Live》,書(shū)中說(shuō)如果想實(shí)現穩定的Excel,推薦使用POI,于是我使用POI生成Excel,穩定性和兼容性都不錯。
六.DBUnit
DBUnit是一個(gè)可以被Ant集成的向數據庫中添加數據和備份數據的一個(gè)類(lèi)庫,配置很方便,因為AppFuse已經(jīng)集成好了,所以使用也很容易。
但是如果你使用EditPlus之類(lèi)的工具手工修改了AppFuse生成的內容,則執行Ant的setup、setup-db或者deploy的任務(wù)時(shí),常常報錯,說(shuō)無(wú)效的格式,這是因為這個(gè)被手工修改的文件再次被AppFuse執行后,它的第一行的文件聲明的前幾個(gè)字母是無(wú)效的,是因為本地的字符集編碼的原因而引起了亂碼,如果把這幾個(gè)無(wú)效的字母去掉,問(wèn)題就解決了。
七.Struts Menu
Struts Menu也是AppFuse的作者開(kāi)發(fā)的一個(gè)開(kāi)源軟件,它可以根據配置文件讀取當前用戶(hù)可以使用的功能菜單,這個(gè)功能是我一直以來(lái)都想要的,我也找到了一些代碼,但實(shí)現的都不如這個(gè)完善,沒(méi)什么好說(shuō)的,使用簡(jiǎn)單,配置容易,很好的解決了我的問(wèn)題。
問(wèn)題是我只使用了AppFuse提供的2個(gè)角色,對于多個(gè)角色的實(shí)驗我還沒(méi)有做。
八.XDoclet
在A(yíng)ppFuse中,使用Xdoclet生成了幾乎一切的配置文件:Struts-config.xml、web.xml、validation.xml、*.hbm.xml等文件,如果使用AppGen的話(huà),還會(huì )生成更多的文件,這一切都是使用Xdoclet實(shí)現的。
問(wèn)題是我在Struts部分提到的,生成的Validation.xml文件中會(huì )多生成一個(gè)“.”,另外在生成資源文件時(shí)也會(huì )多生成一個(gè)“.”,目前我沒(méi)有很好的閱讀這段代碼,不知道是不是Xdoclet的問(wèn)題。
九.Ant
Ant并沒(méi)有什么問(wèn)題,但在執行作者寫(xiě)的Ant任務(wù)的時(shí)候,有一些任務(wù)不能正常執行,比如,運行模擬對象測試的任務(wù),作者也在1.7版本的修復列表中提到以前版本有些ant任務(wù)不能執行,在1.7中修改了一些ant任務(wù),使他們能夠正常的執行了。
實(shí)際上,我們如果使用AppGen進(jìn)行開(kāi)發(fā)的話(huà),使用的任務(wù)一般不超過(guò)8個(gè)。
十.JSTL
JSTL是個(gè)好東西,我常用的有和部分的標簽,但是如果使用JSTL進(jìn)行邏輯判斷,我并沒(méi)有感覺(jué)比使用JSP的代碼塊優(yōu)雅多少。另外,熟悉JSTL也需要一段時(shí)間,我就經(jīng)歷了面對著(zhù)JSP頁(yè)面不知道該怎么寫(xiě)JSTL語(yǔ)法的困境。當然,AppFuse中使用的基本都是JSTL,包括向DisplayTag傳遞顯示的數據,使用的都是JSTL語(yǔ)法,這方面的資料挺多,我參考的是電子工業(yè)出版社出的《JSP2.0技術(shù)》,說(shuō)的很詳細。
十一.Tomcat
你也許會(huì )說(shuō):“Tomcat就不用說(shuō)了吧?”,是的,Tomcat一般都會(huì )使用,但是?D?D?D?D?D?D?D?D?D?D?D?D?DTomcat5和Tomcat4.X對于中文編碼使用了不同的機制,這個(gè)問(wèn)題困擾了我好久,我解決了頁(yè)面上寫(xiě)入漢字顯示亂碼的問(wèn)題,我也曾經(jīng)以為DisplayTag對漢字不能排序,也不能正常分頁(yè)是因為DisplayTag的開(kāi)發(fā)者都是老外,是因為他們沒(méi)有考慮中文的關(guān)系的原因。
直到昨天,我才知道這一切都是因為T(mén)omcat5對漢字編碼的實(shí)現的方式和Tomcat4不一樣的原因,如果感興趣,可以看看這個(gè)帖子:http://www.javaworld.com.tw/jute/post/view?bid=9&id=44042&sty=1&tpg=1&age=0
十二.JavaScript
JavaScript簡(jiǎn)單易學(xué),但想運用自如就不太容易了。AppFuse中嵌入了幾個(gè)js文件,里面包含了許多函數,值得我們好好的研究一下,比如,如果有一個(gè)必填字段沒(méi)有填寫(xiě),AppFuse會(huì )自動(dòng)的聚焦在那個(gè)input上,類(lèi)似的小技巧有很多,你可以自己去翻看。
但AppFuse自帶的JavaScript腳本有一個(gè)Bug,就是當DisplatyTag中沒(méi)有可以顯示的數據時(shí),你用鼠標單擊,它會(huì )報JavaScript錯誤,你仔細研究一下function highlightTableRows(tableId) 就知道了:我的解決辦法是在location.href = link.getAttribute("href");前面添加一行判斷:if (link != null)。
十三.資源文件國際化
對于Struts和DisplayTag都涉及到資源文件國際化AppFuse1.6.1很好的解決了Struts資源映射文件國際化的問(wèn)題,你只需要在對應本國語(yǔ)言的資源文件中寫(xiě)入漢字,Ant中有一項執行native2ascii的任務(wù),AppFuse自動(dòng)的為你進(jìn)行了資源文件的編碼轉換,而對于DisplayTag的資源文件問(wèn)題,還要自己執行native2ascii命令,為了避免每次都輸入一串命令,我用Delphi寫(xiě)了個(gè)小工具,可視化的選擇資源文件,點(diǎn)擊按鈕自動(dòng)執行該命令,底層依賴(lài)于JDK。
經(jīng)過(guò)2個(gè)多月的學(xué)習,我感覺(jué)這個(gè)框架非常不錯,它為我以后的項目開(kāi)發(fā)指出了一個(gè)新的方向,但如果想很熟練的使用這個(gè)框架進(jìn)行開(kāi)發(fā),至少要對以下幾種技術(shù)比較熟練:Struts(或者WebWork、Spring及其他的已經(jīng)整合進(jìn)來(lái)的MVC框架)、Hibernate(或者ibatis)、JSTL,當然其他的技術(shù)至少也要知道一點(diǎn),否則遇到問(wèn)題都不知道出在哪里。
目前我還沒(méi)有解決的問(wèn)題有:
1. 如何在翻頁(yè)的時(shí)候才讀取下面的數據?
2. 怎樣對使用同一個(gè)FormBean的多個(gè)Form進(jìn)行客戶(hù)端校驗?
3. 怎樣優(yōu)化Hibernate的效率?《Hibernate In Action》中提供了多種策略,有些時(shí)候應該使用lazy,有些時(shí)候應該使用outer-join。
4. 在什么時(shí)機生成導出文件?目前我是在查詢(xún)的Action中同時(shí)生成了導出文件,否則,到了下一頁(yè),我就不知道查詢(xún)條件了,當然,如果把拼裝后的HQL存儲在Session或者Hidden中也可以解決這個(gè)問(wèn)題,但是這樣就破壞了DAO的封裝,要把DAO封裝后的HQL發(fā)送給Action,然后發(fā)送的到Web界面層,所以目前我還在猶豫生成導出文件的時(shí)機選擇在哪里?
5. 什么時(shí)候應該自己獲取數據庫連接,執行native SQL?具體需要注意些什么?
6. SiteMesh的模板優(yōu)化?
7. DisplayTag的底層實(shí)現?
每個(gè)問(wèn)題都比較棘手,要一個(gè)一個(gè)解決!
這個(gè)框架的優(yōu)點(diǎn)是:如果熟悉了開(kāi)發(fā)流程,可以大幅度的提高開(kāi)發(fā)速度,如果業(yè)務(wù)不是很復雜,使用AppGen可以生成60%左右的代碼,而且程序可維護性好,因為作者使用了多個(gè)設計模式對各個(gè)層面進(jìn)行了封裝,所以不同的模塊代碼風(fēng)格出奇的一致,有利于開(kāi)發(fā)人員快速上手,也有利于接收其他開(kāi)發(fā)人員遺留的代碼。
本站僅提供存儲服務(wù),所有內容均由用戶(hù)發(fā)布,如發(fā)現有害或侵權內容,請
點(diǎn)擊舉報。