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

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

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

    soufan

      BlogJava :: 首頁 :: 新隨筆 :: 聯系 :: 聚合  :: 管理 ::
      22 隨筆 :: 0 文章 :: 0 評論 :: 0 Trackbacks

    2006年9月3日 #

         摘要: 轉載摘要:本文介紹了 Web Services 的起源和基本原理,分析了在企業(yè)應用中 Web Services 帶來的沖擊和變革,指出了 Web Services ...  閱讀全文
    posted @ 2007-03-04 14:12 soufan 閱讀(220) | 評論 (0)編輯 收藏

    (原)Java 專業(yè)人士必備的書籍和網站列表

    您必備的參考資料

    文檔選項
    ?

    ?

    未顯示需要 JavaScript 的文檔選項


    拓展 Tomcat 應用

    下載 IBM 開源 J2EE 應用服務器 WAS CE 新版本 V1.1


    級別: 初級

    Roy Miller (roy@roywmiller.com), 創(chuàng)始人兼總裁, The Other Road, LLC

    2007 年 1 月 15 日

    對于 Java? 語言開發(fā)人員來說,信息過量是一個真正的問題。每個新入行的程序員都要面臨一個令人畏縮的挑戰(zhàn):要進入的行業(yè)是一個具有海量知識的行業(yè)。要了解的東西簡直太多了。對于有經驗的老手來說,情況只有些微好轉。知識量總在增大,僅僅跟上進度就是一個挑戰(zhàn)。如果有一份專業(yè)人士必備的書籍和網站列表該有多好!本文就是這個列表。它包含了每個專業(yè)的 Java 語言程序員在書架或瀏覽器書簽中必備的最重要的書籍和網站。

    這些都是您書架上必備的書和應該經常使用的 Web 鏈接。時間是一項重要的資源,本文幫您回避那些分心的事情,把時間專注于最有益于您作為Java 語言程序員職業(yè)生涯的信息源。盡管有多少程序員就有多少他們最喜歡的參考資料,但本文收集的這些都是優(yōu)中選優(yōu),來源于我書架上的私家珍藏和許多 Java 專家的推薦。

    我考慮了兩種組織這份參考資料列表的方法。我本可以通過主題領域來組織,這也許很有幫助,但主題列表很快就會變得不實用。相反,我選擇了另一種方法:通過類型來組織,即書籍和 Web 站點。

    總的來講,有經驗的老手們用 Web 站點來跟蹤行業(yè)的走勢。書籍、文章和論文有助于跟上潮流,但它們總體上更適合于基礎學習。極富創(chuàng)造性的書籍偶爾會撼動一兩個基礎性的東西。這樣的書也在本列表之列。

    需要提出的一點警告是,專注于 Java 語言的書籍和 Web 站點數量巨大。您鐘愛的未必在這份列表里。那并不意味著它們不好。它們只是不在這份列表里而已。可能是因為我還不知道它們。也可能是因為我不認為它們能夠算得上是重要資源。不包含一些參考資料是一個評判問題,但如果不這樣的話,您也許就要花幾小時來拖動滾動條,還要花上成千上萬美元來買書。如果您作為一個專業(yè)的 Java 程序員,有一些常用的優(yōu)秀參考資料,一定要讓我知道這些資料。這份列表一直都在更新中,您提出的那些也許就會被收錄進去。

    書籍

    每個程序員都會有一些由于經常被當作專業(yè)資料參閱而磨壞的書。下列書籍應該是 Java 語言程序員的書架上必備的。書很貴,所以我有意將這份列表弄得很短,僅限于重要書籍。

    Thinking in Java (Bruce Eckel)

    Thinking in Java, 3rd edition (Bruce Eckel; Prentice Hall PTR,2002 年)
    Java 編程思想:第3版 (陳昊鵬 等譯; 機械工業(yè)出版社,2005 年)
    Eckel 的書對于學習如何在 Java 語言環(huán)境中使用好面向對象技術極其實用。書中大量的代碼樣例解釋了他所介紹的概念。文字出自一個并不認為 Java 技術總是正確答案的人,所以相當地實用。Eckel 具有多種語言的大量經驗,還有用面向對象方式進行思考的扎實技能。本書將這些技能放到實用的 Java 語言環(huán)境中。他還在寫一本新書,名為 Thinking in Enterprise Java

    Effective Java (Joshua Bloch)

    Effective Java: Programming Language Guide (Joshua Bloch; Addison-Wesley,2001 年)
    Effective Java 中文版 (潘愛民 譯; 機械工業(yè)出版社,2003 年)
    本書是理解優(yōu)秀 Java 程序設計原則的最佳書籍。大多數材料從其他的 “學習 Java ” 的書中根本找不到。例如,Bloch 書中關于覆蓋 equals() 這一章是我讀過的最好的參考資料之一。他也在書中包括了很實用的建議:用接口替代抽象類和靈活使用異常。Bloch 是 Sun 公司 Java 平臺庫的架構師,所以他透徹地了解這門語言。事實上,他編寫了該語言中大量有用的庫。本書必讀!

    The Java Programming Language (Ken Arnold, James Gosling, David Holmes)

    The Java Programming Language (Ken Arnold,James Gosling,David Holmes; Addison-Wesley,2000 年)
    Java 編程語言(第 3 版) (虞萬榮 等譯,中國電力出版社,2003 年)
    這也許是能弄到的最好的 Java 入門讀物。它并不是一個標準規(guī)范,而是一本介紹每門語言特性的可讀書籍。這本書在嚴謹性和教育性方面權衡得很好,能夠讓懂編程的人迅速被 Java 語言(和其豐富的類庫)所吸引。

    Concurrent Programming in Java: Design Principles and Patterns (Doug Lea)

    Concurrent Programming in Java: Design Principles and Patterns, 2nd edition (Doug Lea; Addison-Wesley,1999 年)
    Java 并發(fā)編程—設計原則與模式(第二版) (趙涌 等譯,中國電力出版社,2004 年)
    不是每個開發(fā)人員都需要如此細致地了解并發(fā)性,也不是每個工程師都能達到本書的水準,但卻沒有比本書更好的關于并發(fā)性編程的概述了。如果您對此感興趣,請從這里開始。Lea 是 SUNY 的一名專業(yè)程序員,他的和并發(fā)性有關的作品和想法都包含在了 JDK 5.0 規(guī)范(引自 JSR166)中,所以您大可放心,他所說的關于有效使用 Java 語言的建議是值得一聽的。他是一個很善于溝通的人。

    Expert One-On-One J2EE Design and Development (Rod Johnson)

    Expert One-On-One J2EE Design and Development (Rod Johnson)
    WROX: J2EE 設計開發(fā)編程指南 (魏海萍 譯,電子工業(yè)出版社,2003 年)
    對于剛接觸 J2EE 的人來說,這是唯一的一本如實反映這項技術的書。本書收錄了多年的成功經驗和失敗經驗,不同于其他許多作者,Johnson 樂于將失敗的經驗公諸于眾。J2EE 常常都被過度使用。Johnson 的書能幫您避免這一點。

    Refactoring (Martin Fowler, Kent Beck, John Brant, William Opdyke, Don Roberts)

    Refactoring: Improving the Design of Existing Code (Martin Fowler,Kent Beck,John Brant,William Opdyke,Don Roberts; Addison-Wesley,1999 年)
    重構:改善既有代碼的設計(中文版) (侯捷 等譯,中國電力出版社 ,2003 年)
    Fowler 寫了幾本現已出版的最流行的編程書,包括 Analysis Patterns。他的關于重構 的書是這一主題的基本書籍。重構代碼是被程序員忽略的訓練,但卻是程序員最直觀的想法。重構是在不改變代碼結果的前提下改進現有代碼的設計。這是保持代碼整潔的最佳方式,用這種方法設計的代碼總是很容易修改。什么時候進行重構呢?當代碼“散發(fā)出味道”時。Fowler 的書里滿是 Java 語言代碼的例子。許多 Java 語言集成開發(fā)環(huán)境(IDE)(包括了 IBM 的 Eclipse)都將 Fowler 的重構包含了進去,每一個都使用他的重構名命名,所以熟悉如extract method 等重構方法還是很值得的。

    Design Patterns (Erich Gamma, Richard Helm, Ralph Johnson, John Vlissides)

    Design Patterns: Elements of Reusable Object Oriented Software (Erich Gamma,Richard Helm,Ralph Johnson,John Vlissides; Addison-Wesley,1997 年)
    設計模式:可復用面向對象軟件的基礎 (李英軍 等譯,機械工業(yè)出版社 ,2005 年)
    這是一本在專業(yè)程序員圈子里更為有名的書,基于作者共同的綽號,這本書被認為是 “四人幫(GOF)之書”。模式是思考和解決普通編程問題時可以重用的方式。學習模式是一門學科。使用好模式(或知道什么時候 使用模式)是一項技能。忽略模式則是錯誤的。書中所有的例子都以 C++ 表示,但 Java 語言是從那里誕生的,讓 Java 語言程序員由此聯系到如何在 Java 語言中實現這些模式相對簡單一些。熟悉模式并了解如何使用好模式使編程更加簡單。這使得和其他程序員交流也更簡單,因為在針對通用問題的通用解決方案中,模式是描述解決方案中彼此協作的大量相關編程概念的快捷方式。一些更為通用的方式,如工廠方法 則是普便存在的,甚至存在于 Java 語言本身。關于明智使用模式的這個主題,也可以閱讀 Joshua Kerievsky 的 Refactoring to Patterns,該書稱可以讓代碼來告訴您何時實現模式。

    Patterns of Enterprise Application Architecture (Martin Fowler)

    Patterns of Enterprise Application Architecture (Martin Fowler; Addison-Wesley,2002 年)
    企業(yè)應用架構模式 (王懷民 等譯,機械工業(yè)出版社 ,2004 年)
    比起小型、一次性項目來說,企業(yè)開發(fā)當然代表了更大的挑戰(zhàn)。那并不意味著企業(yè)開發(fā)帶來的所有挑戰(zhàn)都是新挑戰(zhàn)。事實上有些時候,這項開發(fā)已經 是以前完成過的了。Fowler 做了很多個這樣的項目。他的書提到了一些通用解決方案,并提供了關于使用、折中和可選方案的指導。Fowler 在書中包含了一些熟悉的模式,如模型視圖控制器(MVC),他也提供了一些您也許不了解的模式,如處理 Web 站點上特定頁面請求或行為請求的 Page Controller 模式。正如您對待大多數模式一樣,一旦您讀過許多模式,您就會認為 “我已經知道那個模式了” 。也許是這樣,但有一個用來引用模式的通用表達方式還是很有幫助的。在有多個組件(由不同人開發(fā))的大型項目中,該類引用是一項很好的幫助。

    UML Distilled (Martin Fowler)

    UML Distilled: A Brief Guide to the Standard Object Modeling Language (Martin Fowler; Addison-Wesley 2003 年)
    UML精粹:標準對象語言簡明指南(第3版) (徐家福 譯,清華大學出版社 ,2005 年)
    對于專業(yè)的程序員來說,UML 是一門很重要的通用可視化溝通語言,但是它被過度使用和草率地濫用了。您無需對使用 UML 溝通了解太多。Martin 對 UML 的提煉為您提供了最核心的東西。事實上,前后的封頁提供了常規(guī)基礎上可能使用到的所有東西。該書中 UML 例子的代碼都是 Java 代碼。

    Test-Driven Development: By Example (Kent Beck)

    Test-Driven Development: By Example (Kent Beck; Addison-Wesley 2002 年)
    測試驅動開發(fā)(中文版) (崔凱 譯,中國電力出版社 ,2004 年)
    測試優(yōu)先編程將使編程發(fā)生革命性變化,能助您成為更好的程序員。在寫代碼之前編寫測試開始很難,但卻是一項威力強大的技能。通過優(yōu)先編寫測試,可使代碼更加簡單,并確保從一開始它就能工作(Beck 實踐著他提倡的測試優(yōu)先,與人合寫了 JUnit,這是 Java 語言最流行的測試框架)。Beck 的書是權威的參考資料,擴展了的 Money 例子也用 Java 語言寫成。Beck 詳述了如何用測試優(yōu)先進行 思考(這也許是許多程序員首先遇到的障礙)。

    The Pragmatic Programmer: From Journeyman to Master (Andy Hunt and Dave Thomas)

    The Pragmatic Programmer: From Journeyman to Master (Andrew Hunt 和 David Thomas; Addison-Wesley 1999 年)
    程序員修煉之道——從小工到專家 (馬維達 譯,電子工業(yè)出版社 ,2004 年)
    做一個純粹的面向對象開發(fā)人員有其優(yōu)勢所在。在當今復雜的社會中,作為 Java 語言開發(fā)人員,為完成任務常要妥協。Hunt 和 Thomas 探討了如何不將真正重要的東西妥協掉而完成任務。這不是一本關于 Java 語言的書,而是 Java 語言開發(fā)人員重要的思想讀物。例如,我認為沒從“要解決問題,而不是推卸責任”這句忠言中受益的程序員,不能像個自豪的藝術家一樣在他的杰作上簽上大名。

    Peopleware: Productive Projects and Teams (Tom DeMarco and Timothy Lister)

    Peopleware: Productive Projects and Teams (Tom DeMarco,Timothy Lister; Dorset House,1999 年)
    人件(第2版) (UMLChina 翻譯組 譯,清華大學出版社 ,2003 年)
    這份列表中的其他所有書籍都至少和技術有些相關。這本書卻不是。在所有技術行話和首字母縮略詞的海洋中,有時軟件開發(fā)人員和經理們會忘記:是 制造了軟件。DeMarco 和 Lister 向我們提醒了這一事實,也向我們提醒了形成這一區(qū)別的原因。這不是一本關于一門特定編程語言的書籍,但卻是每個 Java 語言程序員都應該讀的書。關于 “累死程序員如何讓經理們適得其反” 還有許多其他的好書,但這是最好的一本。





    回頁首


    Web 站點

    Web 站點的數目浩如煙海,如果您想要消化其中的內容,窮畢生之力也難以全部訪問。包含 Java 語言某方面內容的詳盡的網站列表會大得離譜。下列站點都是可靠、真實的。

    Sun 的 Java 技術站點

    Sun 的 Java 語言站點
    這是 Sun 的 Java 語言主站。作為 Java 語言開發(fā)人員,您會發(fā)現自己頻繁地訪問此站點。下列鏈接特別重要,特別是對新入行的 Java 語言開發(fā)人員:

    • New to Java Center
      New to Java Center
      New to Java Center 存放了許多循序漸進的 Java 技術資源鏈接。如果您剛接觸這門語言,這是一個好的起點。
    • 教程和代碼庫
      Java Tutorial
      這里有大名鼎鼎的 Java Tutorial,以及關于 Java 語言各個方面(例如 Collection)的其他教程。

    IBM developerWorks

    IBM 的 developerWorks
    推銷自己也許有些厚臉皮,但 developerWorks 是一項巨大的資源,收錄了大量 Java 語言工具和技術的教程和文章。其內容從初學者指南到學習這門語言到高級并發(fā)性技術。可以根據主題搜索內容,然后根據類型瀏覽。

    Apache Software Foundation

    Apache Software Foundation
    Apache 站點是許多可重用庫(通用領域)和工具的主頁,這些庫和工具幫助 Java 開發(fā)人員進行開發(fā)。這里的內容全都是開放源碼,所以盡管下載想要的吧!許多極其流行的 Java 語言庫和工具(如 Struts、Ant 和 Tomcat)都始于 Apache 項目。Jakarta 專區(qū)匯聚了大多數新興的 Java 語言材料。

    Eclipse.org

    Eclipse
    有幾個好的 Java 語言集成開發(fā)環(huán)境(IDE)。Eclipse(來自 IBM)是最新的 IDE 之一,它很快成為 Java 語言開發(fā)的首要 IDE。它完全是開源的,這意味著它是免費的。該站包含了學習如何有效使用 Eclipse 的各種參考資料。這里還有關于 Standard Widget Toolkit(SWT)的信息,SWT 是相對于 Swing 來說更加輕量級的選擇。

    Eclipse 插件中心和 Eclipse 插件

    Eclipse 插件中心 Eclipse 插件
    Eclipse 基于插件架構。事實上,插件是 Eclipse 的 Java 語言開發(fā)組件。但有差不多上千個插件,從 Web 開發(fā)的插件到在 Eclipse 環(huán)境中玩游戲的插件。這兩個站點分類列出了大多數插件,可以進行搜索。它們是很棒的資源。如果您想在 Eclipse 開發(fā)環(huán)境中弄點新東西,幸運的話有某個插件可能已經實現,從這兩個站點能找到想要的插件。這兩個站點都允許評論插件,這樣您就可以知道哪些插件好,哪些值得一試。

    JUnit.org

    JUnit.org
    Junit 是 Java 語言中一個基本的單元測試框架。該站點包含了 Junit 最新最棒的版本,外加大量有關測試(Java 語言或者其他語言的)各個層面上(針對桌面應用程序、Web 應用程序、J2EE 應用程序等)的其他資源。如果您想找測試資源,這里就是最佳起點。

    TheServerSide.com

    TheServerSide.com
    如果您要(或將要)從事服務器端 Java 語言的開發(fā),此站點是一處舉足輕重的資源。您可以到這里找到有關 JBoss、J2EE、LDAP、Struts 和大量其他主題的文章,并且都是完全可檢索的。這些文章不僅僅是簡單描述 Java 語言的特征或者支持的庫。它們更進一步地描述了庫的新奇用法(如使用 Jakarta Velocity 作為規(guī)則引擎,而不是模板引擎)。它們也提供了有關 Java 語言現狀的連續(xù)評論(當前的一篇文章是由 Tim Bray 所寫的 Java is boring )。該站點更好的通用功能之一是對 Java 語言工具和產品(應用服務器等)的矩陣式比較。

    Bruce Eckel's MindView, Inc.

    Bruce Eckel's MindView, Inc.
    Eckel 寫了幾本 “用 …… 進行思考” 的書,內容關于 Java 語言、Python 和 C++ ,當我學習 Java 語言時,他的 Thinking in Java 對我尤其有幫助。它很實用并切中要害,在“在 Java 語言環(huán)境中如何面向對象思考”方面具有卓識。您可以從此站點免費下載他所有書籍的電子版。他也寫了許多好文章,并且他把這些文章的鏈接都放到了這里(包括關于 Jython、Java 和 .NET 比較等內容的文章)。

    ONJava.com

    ONJava.com
    O'Reilley 歷年來出版了一些有關編程語言和工具的優(yōu)秀書籍。他們的專注于 Java 語言的網站也不錯。它有些有關各種 Java 語言工具(如 JDOM 和 Hibernate)、Java 平臺(如 J2SE 和 J2EE)不同領域不同部分的文章。全部都可以被檢索到。他們有優(yōu)秀的文章和教程。該站點按主題排列。例如有 Java 和 XML、Java Security、Wireless Java 和 Java SysAdmin。該站點也有到 O'Reilley Learning Lab 的鏈接,在那里您能獲得在線參考資料(Java 語言相關和其他的)。那些不是免費的,但是許多都面向大學認證。因此您可以以一種很方便的方式來學習技能,并得到一些認證。

    java.net

    java.net 社區(qū)
    java.net 社區(qū)有多個“社區(qū)”,有特定于主題的論壇和文章。例如 Java Desktop 社區(qū)有各類與 Java 語言桌面開發(fā)相關的資料。Java Patterns 社區(qū)作為一個門戶,也許對提供 Java 語言的模式資源相當感興趣。還有一個 Java User Groups (JUG) 社區(qū),在那里能找到有關創(chuàng)建、加入和管理一個 JUG 的信息。





    回頁首


    結束語

    任何 “好的”、“關鍵性的” 或者 “重要的” 參考資料列表都注定是不完整的,本文的列表也未能例外。 Java 語言的書籍數目眾多,當然,萬維網也很龐大。除本文所列的參考資料之外,還有很多用于學習 Java 語言的參考資料。但如果您擁有了這里所提到的所有書籍、網站、文章或者教程,您應當已經擁有了一個使您良好開端并助您登堂入室的實用寶庫。

    最后,要成為一個能力日增和高效的 Java 語言開發(fā)人員,方法就是用它工作,動手來嘗試。如果有一個教程詳細介紹了所需創(chuàng)建的軟件的每一部分,您很可能并沒得到多少好處。有時,您可能得走自己的路。在成功地嘗試了一些新的東西之后,您可能想要寫一篇文章、教程或者一本書來分享您所學到的。



    參考資料



    關于作者

    Roy Miller 是一名獨立軟件開發(fā)培訓師、程序員兼作家,他在充滿挑戰(zhàn)、快節(jié)奏的咨詢公司里從事了十多年軟件開發(fā)和項目管理工作。他最初在 Andersen Consulting(現在是 Accenture)公司工作,在那里,他管理團隊實現了許多系統,從主機記帳系統到星形模式數據集市。最近三年來,他在北卡羅來納州 Holly Springs 的 RoleModel Software, Inc. 公司工作,在那里他專業(yè)地運用著 Java 技術,并擔任開發(fā)人員兼 Extreme Programming (XP) 培訓師。他與人合著了 Addison-Wesley XP 系列的 Extreme Programming Applied: Playing to Win 一書,最近他寫了 Managing Software for Growth: Without Fear, Control and the Manufacturing Mindset 一書,來幫助經理和管理層理解:像 XP 這樣的敏捷構建方法為什么比傳統的方法更有效。2003 年,他創(chuàng)辦了自己的公司:The Other Road,該公司幫助其他公司了解如何向 XP 和被他稱為 Extreme Business (XB) 的方法轉換。

    posted @ 2007-01-18 13:39 soufan 閱讀(234) | 評論 (0)編輯 收藏

    1. 簡介


    JasperReport是一個強大的開源報表工具,它可以傳送豐富的報表內容到顯示器、打印機或者PDF、HTML、XLS、CSV、XML文件。它完全使用Java編寫,可以在各種Java應用中用來創(chuàng)建動態(tài)報表內容。它的主要目標是用簡單靈活的方法幫助創(chuàng)建便于打印的分頁文檔。

    JasperReport根據一個xml報表設計文件來組織從JDBC獲得的關系數據庫數據。要用數據填充報表,首先必須編譯報表。編譯xml的報表設計文件是用JasperManager類的compileReport()方法完成的。

    通過編譯,報表設計被加載到一個報表設計對象(net.sf.jasperreports.engine.JasperReport類的實例)中并被序列化然后保存。在應用程序用數據填充報表時使用該序列化文件。實際上,報表編譯完成了報表設計中所有的java表達式的編譯。很多檢查工作在編譯期間進行以確保報表設計的完整性,編譯后的文件是待填充的報表,以方便應用程序用各種數據集來產生不同的報表文檔。

    要填充報表,可以使用JasperManager類的fillReportXXX()方法。這些方法接受一個參數代表報表設計——可以是一個JasperDesign對象,也可以是一個存放該類對象的文件名——還有一個獲得填充報表數據的JDBC連接。報表填充的結果是一個表示待打印文檔的對象(net.sf.jasperreports.engine.JasperPrint類的實例),可以被序列化保存以后繼續(xù)使用,或者傳送給打印機、顯示器,或者導出成PDF、HTML、XLS、CSV或者XML文件。

    2. 報表設計

    一個報表設計表示一個模版用來被JasperReport引擎填充數據并傳送到屏幕、打印機或者Web。數據庫的數據根據報表設計被組織來填充報表以得到待打印的分頁文檔。報表設計都保存到一個特定結構的一個XML文件中,文件結構定義在一個JasperReport引擎可以識別的DTD文件中。然后這些xml文件會被編譯以準備報表填充操作。

    創(chuàng)建一個報表設計(模版),必須按照如下結構編輯一個xml文件:

    <?xml version="1.0"?>
    <!DOCTYPE jasperReport
    PUBLIC "-//JasperReports//DTD Report Design//EN"
    "http://jasperreports.sourceforge.net/dtds/jasperreport.dtd">

    <jasperReport name="name_of_the_report" ... >
    ...
    </jasperReport>

    3. 報表參數

    報表參數是傳遞給報表填充操作的對象的引用,為報表引擎?zhèn)鬟f它無法在數據源中找到的數據是非常有用的。例如,我們可以將登陸執(zhí)行報表填充操作的用戶名傳給引擎,這樣我們可以在報表上顯示制表人或者動態(tài)改變報表的標題。

    一個使用報表參數的重要作用是完成報表的動態(tài)查詢語句,以使報表獲得的數據更加符合要求,這些參數就像報表數據的過濾器。

    在報表中聲明參數非常簡單,只需要指定名稱和類型(java類):

    <parameter name="ReportTitle" class="java.lang.String"/>

    <parameter name="MaxOrderID" class="java.lang.Integer"/>

    <parameter name="SummaryImage" class="java.awt.Image"/>

    可以用兩種方法在查詢語句中使用報表參數:

    1. 就像通常在java.sql.PreparedStatement中使用參數一樣:

    SELECT * FROM Orders WHERE OrderID <= $P{MaxOrderID} ORDER BY ShipCountry

    2. 有時需要用參數來動態(tài)改變SQL查詢的部分語句或者將整個SQL語句作為參數傳給報表,在這種情況下,語法有一點不同,如下:

    SELECT * FROM Orders ORDER BY $P!{OrderByClause}

    還有一些報表內建的系統參數可以直接在表達式中使用:

    REPORT_PARAMETERS_MAP

    REPORT_CONNECTION

    REPORT_DATA_SOURCE

    REPORT_SCRIPTLET

    4. 數據源

    JasperReport只是各種類型的數據源,并提供一個JRDataSource的接口。該有一個缺省的實現類(JRResultSetDataSource class)包裝了ResultSet對象,允許使用任何通過JDBC連接的數據庫。使用JDBC數據源時,即可以通過將數據庫連接傳給報表填充引擎并在報表定義中指定一個SQL查詢語句(參考dtd定義中的<queryString>元素)來提供數據,也可以直接用ResultSet作參數生成JRResultSetDataSource對象來提供數據。

    對于其他的數據源,也不會太麻煩,只需要實現JRDataSource接口來創(chuàng)建自己的數據源類。

    5. 字段

    報表字段提供了唯一映射數據源中數據到報表數據的方式。如果數據源是ResultSet對象,報表字段必須對應ResultSet對象中的列,就是說報表字段必須和對應的列有相同的名字和匹配的類型。

    例如,我們要創(chuàng)建的報表需要用Employees表的數據,該表結構如下:

    Column Name Datatype Length
    --------------------------------------
    EmployeeID int 4
    LastName varchar 20
    FirstName varchar 10
    HireDate datetime 8
    我們可以在報表設計文件中定義如下的字段:

    <field name="EmployeeID" class="java.lang.Integer"/>
    <field name="LastName" class="java.lang.String"/>
    <field name="FirstName" class="java.lang.String"/>
    <field name="HireDate" class="java.util.Date"/>
    如果我們生命一個報表字段在ResultSet中沒有對應的列,則會在運行時拋出異常。當然ResultSet中的列沒有被聲明為報表字段不會影響報表的數據填充,但是他們仍然是可以訪問的。

    6. 表達式

    表達式是JasperReport的一個很強大有用的特性。用表達式可以:聲明報表變量來完成各種計算,為數據分組,指定報表文本字段內容或對其他報表對象的顯示進行更靈活的定制。基本上,所有的報表表達式都是Java表達式,并且可以引用報表字段和報表變量。

    在報表設計的xml文件中有諸多定義表達式的元素:<variableExpression>, <initialValueExpression>, <groupExpression>, <printWhenExpression>, <imageExpression> 和<textFieldExpression>。

    要在在表達式中引用報表字段,字段名必須寫在$F{和}符號之間。例如,如果我們要在一個文本域中連接兩個字段,我們可以像下面定義表達式:

    <textFieldExpression>
    $F{FirstName} + " " + $F{LastName}
    </textFieldExpression>
    表達式可以更復雜:

    <textFieldExpression>
    $F{FirstName} + " " + $F{LastName} + " was hired on " +
    (new SimpleDateFormat("MM/dd/yyyy")).format($F{HireDate}) + "."
    </textFieldExpression>
    要在表達式中引用一個變量,必須將變量名寫在$V{和}符號之間,如下:

    <textFieldExpression>
    "Total quantity : " + $V{QuantitySum} + " kg."
    </textFieldExpression>
    對于報表參數也是同樣的語法,只不過參數名必須寫在$P{和}符號之間:

    <textFieldExpression>
    "Max Order ID is : " + $P{MaxOrderID}
    </textFieldExpression>

    7. 變量

    報表變量是在表達式之前構建的專用對象。變量只聲明一次,而可以在整個報表設計中重復使用,并在對應的表達式中完成大量的計算,從而簡化了報表設計。在表達式中,一個變量可以引用其它變量,但是被引用變量必須在引用變量之前聲明。所以變量的聲明順序對報表設計也是很重要的。

    變量還可以聲明來完成引擎內建計算的求值,如:count、sum、average、lowest、highest、variance等等。一個完成Quantity字段sum計算的變量定義如下:

    <variable name="QuantitySum"

    class="java.lang.Double" calculation="Sum">

    <variableExpression>$F{Quantity}</variableExpression>

    </variable>

    我們還可以通過制定初始化級別來改變計算過程,默認的級別是Report就是變量僅在報表開始處初始化一次,一直到報表結束完成計算。我們可以選擇更低的級別讓變量在每個Page、Column或者Group級別重新初始化。假如我們想計算計算每頁的總數,變量聲明如下:

    <variable name="QuantitySum" class="java.lang.Double"
    resetType="Page" calculation="Sum">
    <variableExpression>$F{Quantity}</variableExpression>
    <initialValueExpression>new Double(0) </initialValueExpression>
    </variable>
    變量將在每一頁的開始處被初始化為0。

    引擎還提供了如下的內建變量可以在表達式中直接使用:

    PAGE_NUMBER
    COLUMN_NUMBER
    REPORT_COUNT
    PAGE_COUNT
    COLUMN_COUNT
    GroupName_COUNT

    8. 報表區(qū)域

    在創(chuàng)建報表模板時,我們需要定義報表區(qū)域的內容和風格。一個完全的報表模板包括如下幾個區(qū)域:<title>, <pageHeader>, <columnHeader>, <groupHeader>, <detail>, <groupFooter>, <columnFoter>, <pageFooter>, <summary>。區(qū)域是報表的重要組成部分,它有指定的高度和寬度,并且可以容納直線、矩形、圖片或者文本域等報表對象。我們用<band>標簽在報表模板xml文件中定義報表區(qū)域的內容和風格。下面是一個PageHeader區(qū)域的定義,它僅僅包含一條直線和一個靜態(tài)文本:

    <pageHeader>
    <band height="30">
    <rectangle>
    <reportElement x="0" y="0" width="555" height="25"/>
    <graphicElement/>
    </rectangle>
    <staticText>
    <reportElement x="0" y="0" width="555" height="25"/>
    <textElement textAlignment="Center">
    <font fontName="Helvetica" size="18"/>
    </textElement>
    <text>Northwind Order List</text>
    </staticText>
    </band>
    </pageHeader>

    9. 分組

    組表示一種分組組織數據的方式。填充報表數據時,JasperReport引擎計算所有定義的分組表達式檢查是否出現組邊界(表達式的值改變),如果遇到組邊界則將<groupFooter>和<groupHeader>報表區(qū)域加入報表。

    報表可以包含任意多的分組,組在報表中的聲明順序很重要,因為組之間相互包含。一個組包含其后聲明組依此類推,一個大的組遇到邊界,所有的子組都將被重新初始化。一個報表組跟其數據分組表達式一起定義,同時還需要定義兩個報表分組區(qū)域:分組頭區(qū)域和分組尾區(qū)域。

    關于分組的詳細信息參考分組的報表示例。

    10. 字體和Unicode支持

    現在你可以用任何語言來創(chuàng)建報表。<font>元素的新屬性允許在Java字體和PDF字體間映射。PDF使用特定的字體集使得以前的JasperReport版本沒有辦法使用它們。新的屬性使用戶可以指定什么PDF字體用來顯示不同的字符集(pdfFontName屬性),什么編碼類型(pdfEncoding屬性)和是否將字體嵌入PDF文檔(isPdfEmbedded)。

    為了簡化字體集的使用,增加了一個新屬性<reportFont>。報表字體是報表級別的字體定義用來作為報表中其他顯示對象的默認字體。因為對國際字符集的支持不知為何被綁定到iText庫,你可以在iText documentation.文當中找到更多關于如何用不同的語言不同的字符集創(chuàng)建PDF文檔的信息。

    11. Scriptlets

    所有的報表顯示數據來自報表變量和報表字段,這些數據可以用報表變量和表達式來處理。

    有時候報表需要對變量進行特殊處理,一些變量可能在報表的某個事件中(報表開始、換頁或者換列)被重新初始化,而且,變量在每次從數據源中獲得數據時(每一行)都被計算。而僅僅用簡單變量表達式無法實現所有的復雜功能,這時就要使用Scriptlet。

    因為Scriptlet主要和報表變量一起工作,完全控制scriptlet的執(zhí)行時機非常重要。JasperReport允許根據報表事件定制Java編碼BEFORE或者AFTER:Report、Page、Column和Group的初始化來執(zhí)行Scriptlet。

    要使用Scriptlet,開發(fā)者只需要通過繼承net.sf.jasperreports.engine.JRAbstractScriptlet或者net.sf.jasperreports.engine.JRDefaultScriptlet來創(chuàng)建Scritplet類。該定制的Scriptlet類會被指定為<jasperReport>的scritpletClass屬性的值。創(chuàng)建Scriptlet時開發(fā)這需要實現或者重載如beforeReportInit(), afterReportInit(), beforePageInit(), afterPageInit(), beforeGroupInit(), afterGroupInit(),等方法。這些方法將在填充數據時被引擎在適當的時候調用。

    有一個叫做REPORT_SCRIPTLET的默認報表參數表示對報表引擎在填充數據時實例化的Scriptlet對象的引用。它可以在整個報表的任何表達式中使用來調用Scriptlet的特定方法使整個報表機制更加靈活。

    12. 子報表
    子報表是報表工具的重要特性,它允許創(chuàng)建更復雜的報表并簡化設計工作。自報表在創(chuàng)建主從報表時特別有用。
    posted @ 2006-12-21 17:13 soufan 閱讀(4092) | 評論 (0)編輯 收藏

    如何在JSP頁面中訪問web.xml中的初始化參數?

    你可以使用預定義的JSF EL變量??initParam來訪問:
    例如,如果你有:
    <context-param>
     <param-name>productId</param-name>
    

    如何從java代碼中訪問web.xml 中的初始化參數?

    你可以使用externalContext的 getInitParameter 方法得到他們.例如 如果你的參數如下:
    <context-param>
     <param-name>connectionString</param-name>
     <param-value>jdbc:oracle:thin:scott/tiger@cartman:1521:O901DB</param-value>
    </context-param>
    

    你可以使用下面代碼訪問connectionString :

    FacesContext?fc?=?FacesContext.getCurrentInstance();
    String?connection?=?fc.getExternalContext().getInitParameter("connectionString");

    ?

    posted @ 2006-12-19 16:25 soufan 閱讀(720) | 評論 (0)編輯 收藏

    在客戶端實現可能很簡單.你可以包裝JSP頁面(或者你想要隱藏的一部分)到一個div中,然后你可以添加更多div,當用戶點擊提交按鈕時這些div出現.這些div可以包含gif動畫和其他內容.
    場景:當用戶點擊按鈕,調用JS函數,該函數隱藏頁面并且顯示"請等待..."div.你可以使用CSS來自定義外觀:
    下面是一個正常工作的例子:
    <%@ taglib uri="http://java.sun.com/jsf/html" prefix="h" %>
    <%@ taglib uri="http://java.sun.com/jsf/core" prefix="f" %>
    <f:loadBundle basename="demo.bundle.Messages" var="Message"/>
     
    <html>
    <head> 
      <title>Input Name Page</title>
      <script>
        function gowait() {
          document.getElementById("main").style.visibility="hidden";
          document.getElementById("wait").style.visibility="visible";
        }
       </script>
        
     </head>
     <body bgcolor="white">
      <f:view>
        <div id="main">
           <h1><h:outputText value="#{Message.inputname_header}"/></h1>
           <h:messages style="color: red"/>
           <h:form id="helloForm">
        
             <h:outputText value="#{Message.prompt}"/>
             <h:inputText id="userName" value="#{GetNameBean.userName}" required="true">
               <f:validateLength minimum="2" maximum="20"/>
             </h:inputText>
             <h:commandButton onclick="gowait()" id="submit" 
                   action="#{GetNameBean.action}" value="Say Hello" />
           </h:form>
        </div>
        <div id="wait" style="visibility:hidden; position: absolute; top: 0; left: 0">
           <table width="100%" height ="300px"> 
             <tr>
               <td align="center" valign="middle">
                 <h2>Please, wait...</h2>
               </td>
             </tr>
           </table>
        </div>
      </f:view>
     </body>
    </html>  
    

    如果你想有一個動畫gif圖片在"請等待..."中,當表單提交后該圖片應該從新加載.因此,再一次指定圖片的id,并且添加經過一段時間延時后重新加載的代碼.下面是個例子:

    <script>
     function gowait() {
       document.getElementById("main").style.visibility="hidden";
       document.getElementById("wait").style.visibility="visible";
       window.setTimeout('showProgress()', 500);
     }
      function showProgress(){ 
       var wg = document.getElementById("waitgif");
       wg.src=wg.src;
     }
    </script>
    ....
    ....
    ....
     
    <img id="waitgif" src="animated.gif">
    

    下面是一個使用action listener 的一個例子.
    添加下面的代碼到backing bean的action listener中:
    public?void?viewPdf(ActionEvent?event)?{
    ?String?filename?=?"filename.pdf";

    ?//?use?your?own?method?that?reads?file?to?the?byte?array
    ?byte[]?pdf?=?getTheContentOfTheFile(filename);?

    ?FacesContext?faces?=?FacesContext.getCurrentInstance();
    ?HttpServletResponse?response?=?(HttpServletResponse)?faces.getExternalContext().getResponse();
    ?response.setContentType("application/pdf");
    ?response.setContentLength(pdf.length);
    ?response.setHeader(?"Content-disposition",?"inline;?filename=\""+fileName+"\"");
    ?try?{
    ??ServletOutputStream?out;
    ??out?=?response.getOutputStream();
    ??out.write(pdf);
    ?}?catch?(IOException?e)?{
    ??e.printStackTrace();
    ?}
    ?faces.responseComplete();
    }
    posted @ 2006-12-19 16:20 soufan 閱讀(491) | 評論 (0)編輯 收藏

    使用backing bean來添加UIComponents 到頁面中?

    下面是一個例子:

    jsp1.jsp:

    <%@taglib uri="http://java.sun.com/jsf/core" prefix="f"%>
    <%@taglib uri="http://java.sun.com/jsf/html" prefix="h"%>
    <%@taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c"%>
    <html>
    <f:view>
    <head>
    <title>jsp1</title>
      <link rel="stylesheet" type="text/css" href="./style.css" title="Style"/>
    </head>
    <body bgcolor="#ffffff">  TESTING...
      <h:form id="form1">
        <h:panelGrid id="panelgridtest" binding="#{jsp1Bean.component}"/>
      </h:form>
    </body>
    </f:view>
    </html>
    

    ?

    Jsp1Bean.java:

    package?test;
    ?
    import?javax.faces.application.Application;
    import?javax.faces.component.UIComponent;
    import?javax.faces.component.UIPanel;
    import?javax.faces.component.UIViewRoot;
    import?javax.faces.component.html.HtmlInputText;
    import?javax.faces.component.html.HtmlOutputText;
    import?javax.faces.context.FacesContext;
    ?
    public?class?Jsp1Bean
    {
    ????UIComponent?component?=?null;
    ????FacesContext?facesContext?=?FacesContext.getCurrentInstance();
    ????UIViewRoot?uIViewRoot?=?facesContext.getViewRoot();
    ????Application?application?=?facesContext.getApplication();
    ?
    ????public?Jsp1Bean()
    ????{
    ????}
    ?
    ????public?UIComponent?getComponent()
    ????{
    ????????if?(component?==?null)
    ????????{
    ????????????component?=?new?UIPanel();
    ????????}
    ????????return?component;
    ????}
    ?
    ????public?void?setComponent(UIComponent?component)
    ????{
    ????????this.component?=?component;
    ????}
    ?
    ?????//initialization?block
    ????{
    ????????try
    ????????{
    ????????????//outputText1
    ????????????HtmlOutputText?outputText1?=?(HtmlOutputText)?
    ??????????????facesContext.getApplication().createComponent(HtmlOutputText.COMPONENT_TYPE);
    ????????????outputText1.setValue("---the?outputText1?value---");
    ????????????//inputText1
    ????????????HtmlInputText?inputText1?=?(HtmlInputText)
    ????????????????facesContext.getApplication().createComponent(HtmlInputText.COMPONENT_TYPE);
    ????????????inputText1.setValue("---the?inputText1?value---");
    ?
    ????????????//add?outputText1?and?inputText1?to?component?("UIPanel")
    ????????????this.getComponent().getChildren().add(outputText1);
    ????????????this.getComponent().getChildren().add(inputText1);
    ????????}
    ????????catch?(java.lang.Throwable?t)
    ????????{
    ????????????System.out.println("java.lang.Throwable?exception?encountered...t.getMessage()="?+?t.getMessage());
    ????????????t.printStackTrace();
    ????????}
    ????}
    ?
    ????public?String?doAction()
    ????{
    ????????return?"submit";
    ????}
    }

    ?

    faces-config.xml:

    <?xml version="1.0" encoding="UTF-8"?>
    <!DOCTYPE faces-config PUBLIC "-//Sun Microsystems, Inc.//DTD JavaServer Faces Config 1.1//EN"
                                  "http://java.sun.com/dtd/web-facesconfig_1_1.dtd">
    <faces-config>
      <navigation-rule>
        <from-view-id>/jsp1</from-view-id>
        <navigation-case>
          <from-action>submit</from-action>
          <to-view-id>/jsp1</to-view-id>
          <redirect/>
        </navigation-case>
      </navigation-rule>
      <managed-bean>
        <managed-bean-name>jsp1Bean</managed-bean-name>
        <managed-bean-class>test.Jsp1Bean</managed-bean-class>
        <managed-bean-scope>session</managed-bean-scope>
      </managed-bean>
    </faces-config>
    
    posted @ 2006-12-19 16:14 soufan 閱讀(239) | 評論 (0)編輯 收藏

    如何在BackBean中使用SelectMany 對象?

    Hi.

    This page was made fast and I hope don't be a nuisance to someboby

    首先我們定義backbean來管理我們的 selectMany "list" :

     <managed-bean>
      <managed-bean-name>Manager</managed-bean-name>
     <managed-bean-class>demo.Manager</managed-bean-class>
      <managed-bean-scope>request</managed-bean-scope>
     <managed-bean>
    

    現在Manager類應該有個 SelectMany object來管理信息;

    ?

    package?demo;
    import?javax.faces.component.UISelectMany;
    import?javax.faces.component.UIComponent;

    class?Manager{
    ???
    ????private?UISelectMany?list;
    ???????

    ??????/*
    ???????*?Here?we?must?create?our?list?of?objects
    ???????*?obviously?we?may?add?a?cycle?"while"?or?"for"?
    ???????*?if?we?wouldn't?know?the?number?of?items
    ???????*/?
    ????public?UIComponentBase?getList(){
    ??????
    ?????????list=new?UISelectMany();
    ?????????SelectItem?item?=?new?SelectItem(?"value",?"Description"?);
    ?????????UISelectItem?uiItem?=?new?UISelectItem();
    ?????????uiItem.setValue(?item?);
    ?????????
    ?????????list.getChildren().add(?uiItem?);
    ??????
    ??????return?list;
    ????}

    ????public?void?setList(UIComponentBase?list){
    ????????this.list=(UISelectMany)list;?
    ????}
    ????
    ???public?void?actionListener(ActionEvent?ev){
    ???????
    ???????Object?obj?[]?=?list.getSelectedValues();
    ???????//?obj?is?an?array?of?string?that?contents?the?values?of?our?selectItems
    ???}

    }

    ?

    JSP頁面的代碼如下:?

    <h:selectManyMenu binding="#{Manager.list}">
    

    如果你想使用CheckBox or a ListBox僅僅在JSF文件中該變標簽就可以了.如 <h:selectManyCheckBox> by either <h:selectManyListbox> or <h:selectManyCheckbox>, it is great!!

    posted @ 2006-12-19 16:12 soufan 閱讀(328) | 評論 (0)編輯 收藏

    How to change destination JSP from within RENDER_RESPONSE phase?

    public?void?
    beforePhase ( PhaseEvent?arg0 )?
    {
    if ( arg0.getPhaseId () ==?PhaseId.RENDER_RESPONSE )
    {
    ?? FacesContext?facesContext?=?arg0.getFacesContext () ;
    ?? ViewHandler?vh??=?facesContext.getApplication () .getViewHandler () ;
    ?? UIViewRoot?newRoot?=?vh.createView ( facesContext,? "/yourNew.jsp" ) ;
    ?? facesContext.setViewRoot ( newRoot ) ;
    }



    How to foward to another JSP from an actionListener ActionEvent

    有兩種方法:

    簡單的方法是在commandlink中添加一個 action attribute? .然后你有一個actionListener?和 an action Attribute, 兩個都是可行的.

    但是你還可以使用下面的代碼:

    String?viewId?=?"/edit.jsp";
    FacesContext?context?=?FacesContext.getCurrentInstance();
    UIViewRoot?view?=?context.getApplication().getViewHandler().createView(context,?viewId);
    view.setViewId(viewId);
    context.setViewRoot(view);
    context.renderResponse();


    如何從java代碼中重定向一個JSF頁面

    示例代碼如下:

    public?static?void?redirectPage(String?szPage)
    {
    ?FacesContext?context?=?FacesContext.getCurrentInstance();
    ?javax.faces.application.Application?app?=?context.getApplication();
    ?UIViewRoot?view?=?app.getViewHandler().createView(context,?szPage);
    ?context.setViewRoot(view);
    ?context.renderResponse();
    }
    posted @ 2006-12-19 16:10 soufan 閱讀(378) | 評論 (0)編輯 收藏

    (轉)
    下面是一個email驗證器的示例: ?

    EmailValidator.java:

    import?javax.faces.application.FacesMessage;
    import?javax.faces.component.UIComponent;
    import?javax.faces.context.FacesContext;
    import?javax.faces.validator.Validator;
    import?javax.faces.validator.ValidatorException;
    ?
    public?class?EmailValidator?implements?Validator?{
    ?
    ????private?String?errorMessage?=?null;
    ?
    ????public?void?setErrorMessage(String?errorMessage)?{?this.errorMessage?=?errorMessage;?}
    ?
    ????public?void?validate(FacesContext?context,?UIComponent?component,?Object?value)?{
    ????????if?(null?==?value)?{
    ????????????return;
    ????????}
    ?
    ????????String?email?=?(String)?value;
    ?
    ????????if?(-1?==?email.indexOf('@',?1)?||?-1?==?email.indexOf('.'))?{
    ????????????if?(errorMessage?!=?null)?{
    ????????????????throw?new?ValidatorException(new?FacesMessage(Tags.eval(errorMessage)));
    ????????????}?else?{
    ????????????????//?use?default?validator?message
    ????????????????throw?new?ValidatorException(null);
    ????????????}
    ????????}
    ????}
    }

    Tags.java:

    import?javax.faces.application.Application;
    import?javax.faces.component.UIComponent;
    import?javax.faces.component.UIInput;
    import?javax.faces.context.FacesContext;
    import?javax.faces.el.MethodBinding;
    import?javax.faces.el.ValueBinding;
    import?javax.faces.event.ValueChangeEvent;
    import?javax.faces.webapp.ConverterTag;
    import?javax.faces.webapp.UIComponentTag;
    ?
    public?class?Tags?{
    ???//?Converter?Tags?and?Validator?Tags?helper?methods
    ????public?static?String?eval(String?expression)?{
    ????????if?(expression?!=?null?&&?UIComponentTag.isValueReference(expression))?{
    ????????????FacesContext?context?=?FacesContext.getCurrentInstance();
    ????????????Application?app?=?context.getApplication();
    ????????????ValueBinding?vb?=?app.createValueBinding(expression);
    ????????????return?""?+?vb.getValue(context);
    ????????}?else?{
    ????????????return?expression;
    ????????}
    ????}
    }

    ?

    ?

    ?

    EmailValidatorTag.java:

    import?javax.faces.component.UIComponent;
    import?javax.faces.validator.Validator;
    import?javax.faces.webapp.ValidatorTag;
    import?javax.servlet.jsp.JspException;
    ?
    public?class?EmailValidatorTag?extends?ValidatorTag?{
    ?
    ????private?String?errorMessage?=?null;
    ?
    ????public?EmailValidatorTag()?{
    ????????setValidatorId("Email");
    ????}
    ?
    ????public?void?setErrorMessage(String?errorMessage)?{?this.errorMessage?=?errorMessage;?}
    ?
    ????public?Validator?createValidator()?throws?JspException?{
    ????????EmailValidator?validator?=?(EmailValidator)?super.createValidator();
    ????????validator.setErrorMessage(errorMessage);
    ?
    ????????return?validator;
    ????}
    ?
    ????public?void?release()?{
    ????????errorMessage?=?null;
    ????}
    }

    ?

    ?

    faces-config.xml:

    <validator>
        <validator-id>Email</validator-id>
        <validator-class>EmailValidator</validator-class>
    </validator>
    

    mytags.tld:

    <?xml version="1.0" encoding="UTF-8"?>
     
    <!DOCTYPE taglib PUBLIC
        "-//Sun Microsystems, Inc.//DTD JSP Tag Library 1.2//EN"
        "http://java.sun.com/dtd/web-jsptaglibrary_1_2.dtd">
     
    <taglib>
     
    <tlibversion>1.0</tlibversion>
    <jspversion>1.2</jspversion>
    <shortname>mytags</shortname>
    <uri>mytags</uri>
     
    <tag>
        <name>validateEmail</name>
        <tagclass>EmailValidatorTag</tagclass>
     
        <attribute>
            <name>errorMessage</name>
            <description>message if a validation error occurs</description>
        </attribute>
    </tag>
     
    </taglib>
    

    ?

    mypage.jsp:

    <h:inputText id="email" required="true">
        <mytags:validateEmail errorMessage="#{bean.message}"/>
    </h:inputText>
    
    posted @ 2006-12-19 16:02 soufan 閱讀(236) | 評論 (0)編輯 收藏

    如何為每一個錯誤消息顯示一個圖片

    使用CSS style來實現該功能.例如,你有如下的代碼來顯示消息:
    <div align="center">
     <h:messages id="errMsgs" styleClass="errorFeedback" layout="table" />
    </div>

    ??errorFeedback style class 可能是下面的代碼:

    .errorFeedback {
     color: black;
     vertical-align: middle;
     background-image: url(/AccountSetup/images/warning_feedback.gif);
     background-repeat: no-repeat;
     background-position: left top;
     font-family: Verdana, Arial, Helvetica, sans-serif;
     font-weight: bold;
     line-height: 18px;
     font-size: 10pt;
     text-align: left;
     text-indent: 22px;}
    
    posted @ 2006-12-19 15:59 soufan 閱讀(173) | 評論 (0)編輯 收藏

    (轉) Filter和Servlet中如何訪問FacesContext?

    ?

    在?Faces realm外,例如 在 ?filter 或者servlet中,當 FacesContent.getCurrentInstance() 返回null時候,你可以使用FacesContextFactory來得到FacesContext,下面是一個示例.


    //?You?need?an?inner?class?to?be?able?to?call?FacesContext.setCurrentInstance
    //?since?it's?a?protected?method
    private?abstract?static?class?InnerFacesContext?extends?FacesContext
    {
    ??protected?static?void?setFacesContextAsCurrentInstance(FacesContext?facesContext)?{
    ????FacesContext.setCurrentInstance(facesContext);
    ??}
    }

    private?FacesContext?getFacesContext(ServletRequest?request,?ServletResponse?response)?{
    ??//?Try?to?get?it?first
    ??FacesContext?facesContext?=?FacesContext.getCurrentInstance();
    ??if?(facesContext?!=?null)?return?facesContext;

    ??FacesContextFactory?contextFactory?=?(FacesContextFactory)FactoryFinder.getFactory(FactoryFinder.FACES_CONTEXT_FACTORY);
    ??LifecycleFactory?lifecycleFactory?=?(LifecycleFactory)FactoryFinder.getFactory(FactoryFinder.LIFECYCLE_FACTORY);
    ??Lifecycle?lifecycle?=?lifecycleFactory.getLifecycle(LifecycleFactory.DEFAULT_LIFECYCLE);

    ??//?Either?set?a?private?member?servletContext?=?filterConfig.getServletContext();
    ??//?in?you?filter?init()?method?or?set?it?here?like?this:
    ??//?ServletContext?servletContext?=?((HttpServletRequest)request).getSession().getServletContext();
    ??//?Note?that?the?above?line?would?fail?if?you?are?using?any?other?protocol?than?http

    ??//?Doesn't?set?this?instance?as?the?current?instance?of?FacesContext.getCurrentInstance
    ??facesContext?=?contextFactory.getFacesContext(servletContext,?request,?response,?lifecycle);

    ??//?Set?using?our?inner?class
    ??InnerFacesContext.setFacesContextAsCurrentInstance(facesContext);

    ??//?set?a?new?viewRoot,?otherwise?context.getViewRoot?returns?null
    ??UIViewRoot?view?=?facesContext.getApplication().getViewHandler().createView(facesContext,?"yourOwnID");
    facesContext.setViewRoot(view);

    ??return?facesContext;
    }
    posted @ 2006-12-19 15:52 soufan 閱讀(425) | 評論 (0)編輯 收藏

    J2EE 全面簡介(轉載)

    本文從五個方面對J2EE進行了比較全面的介紹。從J2EE的概念說起,到它的優(yōu)勢,到J2EE典型的四層模型,和它的框架結構,最后是J2EE十三種核心技術的一個簡介。本文分門別類的對J2EE中的服務,組件,層次,容器,API都做了比較詳細的介紹,相信看完此文,讀者會對J2EE有一個更清晰的認識。

    J2EE的概念

    目前,Java 2平臺有3個版本,它們是適用于小型設備和智能卡的Java 2平臺Micro版(Java 2 Platform Micro Edition,J2ME)、適用于桌面系統的Java 2平臺標準版(Java 2 Platform Standard Edition,J2SE)、適用于創(chuàng)建服務器應用程序和服務的Java 2平臺企業(yè)版(Java 2 Platform Enterprise Edition,J2EE)。

    J2EE是一種利用Java 2平臺來簡化企業(yè)解決方案的開發(fā)、部署和管理相關的復雜問題的體系結構。J2EE技術的基礎就是核心Java平臺或Java 2平臺的標準版,J2EE不僅鞏固了標準版中的許多優(yōu)點,例如"編寫一次、隨處運行"的特性、方便存取數據庫的JDBC API、CORBA技術以及能夠在Internet應用中保護數據的安全模式等等,同時還提供了對 EJB(Enterprise JavaBeans)、Java Servlets API、JSP(Java Server Pages)以及XML技術的全面支持。其最終目的就是成為一個能夠使企業(yè)開發(fā)者大幅縮短投放市場時間的體系結構。

    J2EE體系結構提供中間層集成框架用來滿足無需太多費用而又需要高可用性、高可靠性以及可擴展性的應用的需求。通過提供統一的開發(fā)平臺,J2EE降低了開發(fā)多層應用的費用和復雜性,同時提供對現有應用程序集成強有力支持,完全支持Enterprise JavaBeans,有良好的向導支持打包和部署應用,添加目錄支持,增強了安全機制,提高了性能。





    回頁首


    J2EE的優(yōu)勢

    J2EE為搭建具有可伸縮性、靈活性、易維護性的商務系統提供了良好的機制:

    1. 保留現存的IT資產: 由于企業(yè)必須適應新的商業(yè)需求,利用已有的企業(yè)信息系統方面的投資,而不是重新制定全盤方案就變得很重要。這樣,一個以漸進的(而不是激進的,全盤否定的)方式建立在已有系統之上的服務器端平臺機制是公司所需求的。J2EE架構可以充分利用用戶原有的投資,如一些公司使用的BEA Tuxedo、IBM CICS, IBM Encina,、Inprise VisiBroker 以及Netscape Application Server。這之所以成為可能是因為J2EE擁有廣泛的業(yè)界支持和一些重要的'企業(yè)計算'領域供應商的參與。每一個供應商都對現有的客戶提供了不用廢棄已有投資,進入可移植的J2EE領域的升級途徑。由于基于J2EE平臺的產品幾乎能夠在任何操作系統和硬件配置上運行,現有的操作系統和硬件也能被保留使用。
    2. 高效的開發(fā): J2EE允許公司把一些通用的、很繁瑣的服務端任務交給中間件供應商去完成。這樣開發(fā)人員可以集中精力在如何創(chuàng)建商業(yè)邏輯上,相應地縮短了開發(fā)時間。高級中間件供應商提供以下這些復雜的中間件服務:
      • 狀態(tài)管理服務 -- 讓開發(fā)人員寫更少的代碼,不用關心如何管理狀態(tài),這樣能夠更快地完成程序開發(fā)。
      • 持續(xù)性服務 -- 讓開發(fā)人員不用對數據訪問邏輯進行編碼就能編寫應用程序,能生成更輕巧,與數據庫無關的應用程序,這種應用程序更易于開發(fā)與維護。
      • 分布式共享數據對象CACHE服務 -- 讓開發(fā)人員編制高性能的系統,極大提高整體部署的伸縮性。
    3. 支持異構環(huán)境: J2EE能夠開發(fā)部署在異構環(huán)境中的可移植程序。基于J2EE的應用程序不依賴任何特定操作系統、中間件、硬件。因此設計合理的基于J2EE的程序只需開發(fā)一次就可部署到各種平臺。這在典型的異構企業(yè)計算環(huán)境中是十分關鍵的。J2EE標準也允許客戶訂購與J2EE兼容的第三方的現成的組件,把他們部署到異構環(huán)境中,節(jié)省了由自己制訂整個方案所需的費用。
    4. 可伸縮性: 企業(yè)必須要選擇一種服務器端平臺,這種平臺應能提供極佳的可伸縮性去滿足那些在他們系統上進行商業(yè)運作的大批新客戶。基于J2EE平臺的應用程序可被部署到各種操作系統上。例如可被部署到高端UNIX與大型機系統,這種系統單機可支持64至256個處理器。(這是NT服務器所望塵莫及的)J2EE領域的供應商提供了更為廣泛的負載平衡策略。能消除系統中的瓶頸,允許多臺服務器集成部署。這種部署可達數千個處理器,實現可高度伸縮的系統,滿足未來商業(yè)應用的需要。
    5. 穩(wěn)定的可用性: 一個服務器端平臺必須能全天候運轉以滿足公司客戶、合作伙伴的需要。因為INTERNET是全球化的、無處不在的,即使在夜間按計劃停機也可能造成嚴重損失。若是意外停機,那會有災難性后果。J2EE部署到可靠的操作環(huán)境中,他們支持長期的可用性。一些J2EE部署在WINDOWS環(huán)境中,客戶也可選擇健壯性能更好的操作系統如Sun Solaris、IBM OS/390。最健壯的操作系統可達到99.999%的可用性或每年只需5分鐘停機時間。這是實時性很強商業(yè)系統理想的選擇。




    回頁首


    J2EE 的四層模型

    J2EE使用多層的分布式應用模型,應用邏輯按功能劃分為組件,各個應用組件根據他們所在的層分布在不同的機器上。事實上,sun設計J2EE的初衷正是為了解決兩層模式(client/server)的弊端,在傳統模式中,客戶端擔當了過多的角色而顯得臃腫,在這種模式中,第一次部署的時候比較容易,但難于升級或改進,可伸展性也不理想,而且經常基于某種專有的協議?D?D通常是某種數據庫協議。它使得重用業(yè)務邏輯和界面邏輯非常困難。現在J2EE 的多層企業(yè)級應用模型將兩層化模型中的不同層面切分成許多層。一個多層化應用能夠為不同的每種服務提供一個獨立的層,以下是 J2EE 典型的四層結構:

    • 運行在客戶端機器上的客戶層組件
    • 運行在J2EE服務器上的Web層組件
    • 運行在J2EE服務器上的業(yè)務邏輯層組件
    • 運行在EIS服務器上的企業(yè)信息系統(Enterprise information system)層軟件


    J2EE應用程序組件
    J2EE應用程序是由組件構成的.J2EE組件是具有獨立功能的軟件單元,它們通過相關的類和文件組裝成J2EE應用程序,并與其他組件交互。J2EE說明書中定義了以下的J2EE組件:

    • 應用客戶端程序和applets是客戶層組件.
    • Java Servlet和JavaServer Pages(JSP)是web層組件.
    • Enterprise JavaBeans(EJB)是業(yè)務層組件.

    客戶層組件
    J2EE應用程序可以是基于web方式的,也可以是基于傳統方式的.

    web 層組件
    J2EE web層組件可以是JSP 頁面或Servlets.按照J2EE規(guī)范,靜態(tài)的HTML頁面和Applets不算是web層組件。

    正如下圖所示的客戶層那樣,web層可能包含某些 JavaBean 對象來處理用戶輸入,并把輸入發(fā)送給運行在業(yè)務層上的enterprise bean 來進行處理。



    業(yè)務層組件
    業(yè)務層代碼的邏輯用來滿足銀行,零售,金融等特殊商務領域的需要,由運行在業(yè)務層上的enterprise bean 進行處理. 下圖表明了一個enterprise bean 是如何從客戶端程序接收數據,進行處理(如果必要的話), 并發(fā)送到EIS 層儲存的,這個過程也可以逆向進行。

    有三種企業(yè)級的bean: 會話(session) beans, 實體(entity) beans, 和消息驅動(message-driven) beans. 會話bean 表示與客戶端程序的臨時交互. 當客戶端程序執(zhí)行完后, 會話bean 和相關數據就會消失. 相反, 實體bean 表示數據庫的表中一行永久的記錄. 當客戶端程序中止或服務器關閉時, 就會有潛在的服務保證實體bean 的數據得以保存.消息驅動 bean 結合了會話bean 和 JMS的消息監(jiān)聽器的特性, 允許一個業(yè)務層組件異步接收JMS 消息.



    企業(yè)信息系統層
    企業(yè)信息系統層處理企業(yè)信息系統軟件包括企業(yè)基礎建設系統例如企業(yè)資源計劃 (ERP), 大型機事務處理, 數據庫系統,和其它的遺留信息系統. 例如,J2EE 應用組件可能為了數據庫連接需要訪問企業(yè)信息系統





    回頁首


    J2EE 的結構

    這種基于組件,具有平臺無關性的J2EE 結構使得J2EE 程序的編寫十分簡單,因為業(yè)務邏輯被封裝成可復用的組件,并且J2EE 服務器以容器的形式為所有的組件類型提供后臺服務. 因為你不用自己開發(fā)這種服務, 所以你可以集中精力解決手頭的業(yè)務問題.

    容器和服務
    容器設置定制了J2EE服務器所提供得內在支持,包括安全,事務管理,JNDI(Java Naming and Directory Interface)尋址,遠程連接等服務,以下列出最重要的幾種服務:

    • J2EE安全(Security)模型可以讓你配置 web 組件或enterprise bean ,這樣只有被授權的用戶才能訪問系統資源. 每一客戶屬于一個特別的角色,而每個角色只允許激活特定的方法。你應在enterprise bean的布置描述中聲明角色和可被激活的方法。由于這種聲明性的方法,你不必編寫加強安全性的規(guī)則。
    • J2EE 事務管理(Transaction Management)模型讓你指定組成一個事務中所有方法間的關系,這樣一個事務中的所有方法被當成一個單一的單元. 當客戶端激活一個enterprise bean中的方法,容器介入一管理事務。因有容器管理事務,在enterprise bean中不必對事務的邊界進行編碼。要求控制分布式事務的代碼會非常復雜。你只需在布置描述文件中聲明enterprise bean的事務屬性,而不用編寫并調試復雜的代碼。容器將讀此文件并為你處理此enterprise bean的事務。
    • JNDI 尋址(JNDI Lookup)服務向企業(yè)內的多重名字和目錄服務提供了一個統一的接口,這樣應用程序組件可以訪問名字和目錄服務.
    • J2EE遠程連接(Remote Client Connectivity)模型管理客戶端和enterprise bean間的低層交互. 當一個enterprise bean創(chuàng)建后, 一個客戶端可以調用它的方法就象它和客戶端位于同一虛擬機上一樣.
    • 生存周期管理(Life Cycle Management)模型管理enterprise bean的創(chuàng)建和移除,一個enterprise bean在其生存周期中將會歷經幾種狀態(tài)。容器創(chuàng)建enterprise bean,并在可用實例池與活動狀態(tài)中移動他,而最終將其從容器中移除。即使可以調用enterprise bean的create及remove方法,容器也將會在后臺執(zhí)行這些任務。
    • 數據庫連接池(Database Connection Pooling)模型是一個有價值的資源。獲取數據庫連接是一項耗時的工作,而且連接數非常有限。容器通過管理連接池來緩和這些問題。enterprise bean可從池中迅速獲取連接。在bean釋放連接之可為其他bean使用。

    容器類型
    J2EE應用組件可以安裝部署到以下幾種容器中去:

    • EJB 容器管理所有J2EE 應用程序中企業(yè)級bean 的執(zhí)行. enterprise bean 和它們的容器運行在J2EE 服務器上.
    • Web 容器管理所有J2EE 應用程序中JSP頁面和Servlet組件的執(zhí)行. Web 組件和它們的容器運行在J2EE 服務器上.
    • 應用程序客戶端容器管理所有J2EE應用程序中應用程序客戶端組件的執(zhí)行. 應用程序客戶端和它們的容器運行在J2EE 服務器上.
    • Applet 容器是運行在客戶端機器上的web瀏覽器和 Java 插件的結合.






    回頁首


    J2EE的核心API與組件

    J2EE平臺由一整套服務(Services)、應用程序接口(APIs)和協議構成,它對開發(fā)基于Web的多層應用提供了功能支持,下面對J2EE中的13種技術規(guī)范進行簡單的描述(限于篇幅,這里只能進行簡單的描述):

    1. JDBC(Java Database Connectivity): JDBC API為訪問不同的數據庫提供了一種統一的途徑,象ODBC一樣,JDBC對開發(fā)者屏蔽了一些細節(jié)問題,另外,JDCB對數據庫的訪問也具有平臺無關性。
    2. JNDI(Java Name and Directory Interface): JNDI API被用于執(zhí)行名字和目錄服務。它提供了一致的模型來存取和操作企業(yè)級的資源如DNS和LDAP,本地文件系統,或應用服務器中的對象。
    3. EJB(Enterprise JavaBean): J2EE技術之所以贏得某體廣泛重視的原因之一就是EJB。它們提供了一個框架來開發(fā)和實施分布式商務邏輯,由此很顯著地簡化了具有可伸縮性和高度復雜的企業(yè)級應用的開發(fā)。EJB規(guī)范定義了EJB組件在何時如何與它們的容器進行交互作用。容器負責提供公用的服務,例如目錄服務、事務管理、安全性、資源緩沖池以及容錯性。但這里值得注意的是,EJB并不是實現J2EE的唯一途徑。正是由于J2EE的開放性,使得有的廠商能夠以一種和EJB平行的方式來達到同樣的目的。
    4. RMI(Remote Method Invoke): 正如其名字所表示的那樣,RMI協議調用遠程對象上方法。它使用了序列化方式在客戶端和服務器端傳遞數據。RMI是一種被EJB使用的更底層的協議。
    5. Java IDL/CORBA: 在Java IDL的支持下,開發(fā)人員可以將Java和CORBA集成在一起。他們可以創(chuàng)建Java對象并使之可在CORBA ORB中展開, 或者他們還可以創(chuàng)建Java類并作為和其它ORB一起展開的CORBA對象的客戶。后一種方法提供了另外一種途徑,通過它Java可以被用于將你的新的應用和舊的系統相集成。
    6. JSP(Java Server Pages): JSP頁面由HTML代碼和嵌入其中的Java代碼所組成。服務器在頁面被客戶端所請求以后對這些Java代碼進行處理,然后將生成的HTML頁面返回給客戶端的瀏覽器。
    7. Java Servlet: Servlet是一種小型的Java程序,它擴展了Web服務器的功能。作為一種服務器端的應用,當被請求時開始執(zhí)行,這和CGI Perl腳本很相似。Servlet提供的功能大多與JSP類似,不過實現的方式不同。JSP通常是大多數HTML代碼中嵌入少量的Java代碼,而servlets全部由Java寫成并且生成HTML。
    8. XML(Extensible Markup Language): XML是一種可以用來定義其它標記語言的語言。它被用來在不同的商務過程中共享數據。XML的發(fā)展和Java是相互獨立的,但是,它和Java具有的相同目標正是平臺獨立性。通過將Java和XML的組合,您可以得到一個完美的具有平臺獨立性的解決方案。
    9. JMS(Java Message Service): MS是用于和面向消息的中間件相互通信的應用程序接口(API)。它既支持點對點的域,有支持發(fā)布/訂閱(publish/subscribe)類型的域,并且提供對下列類型的支持:經認可的消息傳遞,事務型消息的傳遞,一致性消息和具有持久性的訂閱者支持。JMS還提供了另一種方式來對您的應用與舊的后臺系統相集成。
    10. JTA(Java Transaction Architecture): JTA定義了一種標準的API,應用系統由此可以訪問各種事務監(jiān)控。
    11. JTS(Java Transaction Service): JTS是CORBA OTS事務監(jiān)控的基本的實現。JTS規(guī)定了事務管理器的實現方式。該事務管理器是在高層支持Java Transaction API (JTA)規(guī)范,并且在較底層實現OMG OTS specification的Java映像。JTS事務管理器為應用服務器、資源管理器、獨立的應用以及通信資源管理器提供了事務服務。
    12. JavaMail: JavaMail是用于存取郵件服務器的API,它提供了一套郵件服務器的抽象類。不僅支持SMTP服務器,也支持IMAP服務器。
    13. JTA(JavaBeans Activation Framework): JavaMail利用JAF來處理MIME編碼的郵件附件。MIME的字節(jié)流可以被轉換成Java對象,或者轉換自Java對象。大多數應用都可以不需要直接使用JAF。
    posted @ 2006-12-18 17:35 soufan 閱讀(135) | 評論 (0)編輯 收藏

    from:http://www.javaeye.com/topic/9706

    數據庫對象的緩存策略

    前言
    本文探討Jive(曾經開源的Java論壇)和Hibernate(Java開源持久層)的數據庫對象的緩存策略,并闡述作者本人的Lightor(Java開源持久層)采用的數據庫對象緩存策略。
    本文的探討基于以前開源的Jive代碼,Hibernate2.1.7源碼,和作者本人的Lightor代碼。
    本文用ID (Identifier的縮寫)來代表數據記錄的關鍵字。
    數據對象查詢一般分為兩種:條件查詢,返回一個滿足條件的數據對象列表; ID查詢,返回ID對應的數據對象。
    本文主要探討“條件查詢”和“ID查詢”這兩種情況的緩存策略。
    本文只探討一個JVM內的數據緩存策略,不涉及分布式緩存;本文只探討對應單表的數據對象的緩存,不涉及關聯表對象的情況。

    一、Jive的緩存策略
    1.Jive的緩存策略的過程描述:
    (1)條件查詢的時候,Jive用 select id from table_name where …. (只選擇ID字段)這樣的SQL語句查詢數據庫,來獲得一個ID列表。
    (2) Jive根據ID列表中的每個ID,首先查看緩存中是否存在對應ID的數據對象:如果存在,那么直接取出,加入到 結果列表中;如果不存在,那么通過一條select * from table_name where id = {ID value} 這樣的SQL查詢數據庫,取出對應的數據對象,放入到結果列表,并把這個數據對象按照ID放入到緩存中。
    (3) ID查詢的時候,Jive執(zhí)行類似第(2)步的過程,先從緩存中查找該ID,查不到,再查詢數據庫,然后把結果放入到緩存。
    (4) 刪除、更新、增加數據的時候,同時更新緩存。
    2.Jive緩存策略的優(yōu)點:
    (1) ID查詢的時候,如果該ID已經存在于緩存中,那么可以直接取出。節(jié)省了一條數據庫查詢。
    (2) 當多次條件查詢的結果集相交的情況下,交集里面的數據對象不用重復從數據庫整個獲取,直接從緩存中獲取即可。
    比如,第一次查詢的ID列表為{1, 2},然后根據ID列表的ID從數據庫中一個一個取出數據對象,結果集為{a(id = 1), b(id = 2)}。
    下一次查詢的ID列表為{2, 3},由于ID = 2的數據對象已經存在于緩存中,那么只要從數據庫中取出ID = 3的數據對象即可。
    3.Jive緩存策略的缺點:
    (1) 在根據條件查找數據對象列表的過程中,DAO的第(1)步用來獲得ID列表的那一次數據庫查詢,是必不可少的。
    (2) 如果第(1)步返回的ID列表中有n個ID,在最壞的命中率(緩存中一個對應ID都沒有)情況下,Jive還要再查詢n次數據庫。最壞情況下,共需要n + 1數據庫查詢。

    二、Hibernate的二級緩存策略
    Hibernate用Session類包裝了數據庫連接從打開到關閉的過程。
    Session內部維護一個數據對象集合,包括了本Session內選取的、操作的數據對象。這稱為Session內部緩存,是Hibernate的第一級最快緩存,屬于Hibernate的既定行為,不需要進行配置(也沒有辦法配置 :-)。
    Session的生命期很短,存在于Session內部的第一級最快緩存的生命期當然也很短,命中率自然也很低。當然,這個Session內部緩存的主要作用是保持Session內部數據狀態(tài)同步。
    如果需要跨Session的命中率較高的全局緩存,那么必須對Hibernate進行二級緩存配置。一般來說,同樣數據類型(Class)的數據對象,共用一個二級緩存(或其中的同一塊)。
    1.Hibernate二級緩存策略的過程描述:
    (1)條件查詢的時候,總是發(fā)出一條select * from table_name where …. (選擇所有字段)這樣的SQL語句查詢數據庫,一次獲得所有的數據對象。
    (2) 把獲得的所有數據對象根據ID放入到第二級緩存中。
    (3) 當Hibernate根據ID訪問數據對象的時候,首先從Session一級緩存中查;查不到,如果配置了二級緩存,那么從二級緩存中查;查不到,再查詢數據庫,把結果按照ID放入到緩存。
    (4) 刪除、更新、增加數據的時候,同時更新緩存。

    2.Hibernate二級緩存策略的優(yōu)點:
    (1) 具有Jive緩存策略同樣的第(1)條優(yōu)點:ID查詢的時候,如果該ID已經存在于緩存中,那么可以直接取出。節(jié)省了一條數據庫查詢。
    (2) 不具有Jive緩存策略的第(2)條缺點,即hibernate不會有最壞情況下的 n + 1次數據庫查詢。
    3.Hibernate二級緩存策略的缺點:
    (1) 同Jive緩存策略的第(1)條缺點一樣,條件查詢的時候,第(1)步的數據庫查詢語句是不可少的。而且Hibernate選擇所有的字段,比只選擇ID字段花費的時間和空間都多。
    (2) 不具備Jive緩存策略的第(2)條優(yōu)點。條件查詢的時候,必須把數據庫對象從數據庫中整個取出,即使該數據庫的ID已經存在于緩存中。

    三、Hibernate的Query緩存策略
    可以看到,Jive緩存和Hibernate的二級緩存策略,都只是針對于ID查詢的緩存策略,對于條件查詢則毫無作用。(盡管Jive緩存的第(2)個優(yōu)點,能夠避免重復從數據庫獲取同一個ID對應的數據對象,但select id from …這條數據庫查詢是每次條件查詢都必不可少的)。
    為此,Hibernate提供了針對條件查詢的Query緩存。
    1.Hibernate的Query緩存策略的過程描述:
    (1) 條件查詢的請求一般都包括如下信息:SQL, SQL需要的參數,記錄范圍(起始位置rowStart,最大記錄個數maxRows),等。
    (2) Hibernate首先根據這些信息組成一個Query Key,根據這個Query Key到Query緩存中查找對應的結果列表。如果存在,那么返回這個結果列表;如果不存在,查詢數據庫,獲取結果列表,把整個結果列表根據Query Key放入到Query緩存中。
    (3) Query Key中的SQL涉及到一些表名,如果這些表的任何數據發(fā)生修改、刪除、增加等操作,這些相關的Query Key都要從緩存中清空。
    2.Hibernate的Query緩存策略的優(yōu)點
    (1) 條件查詢的時候,如果Query Key已經存在于緩存,那么不需要再查詢數據庫。命中的情況下,一次數據庫查詢也不需要。
    3.Hibernate的Query緩存策略的缺點
    (1) 條件查詢涉及到的表中,如果有任何一條記錄增加、刪除、或改變,那么緩存中所有和該表相關的Query Key都會失效。
    比如,有這樣幾組Query Key,它們的SQL里面都包括table1。
    SQL = select * from table1 where c1 = ? …., parameter = 1, rowStart = 11, maxRows = 20.
    SQL = select * from table1 where c1 = ? …., parameter = 1, rowStart = 21, maxRows = 20.
    SQL = select * from table1 where c1 = ? ….., parameter = 2, rowStart = 11, maxRows = 20.
    SQL = select * from table1 where c1 = ? ….., parameter = 2, rowStart = 11, maxRows = 20.
    SQL = select * from table1 where c2 = ? …., parameter = ‘abc’, rowStart = 11, maxRows = 20.

    當table1的任何數據對象(任何字段)改變、增加、刪除的時候,這些Query Key對應的結果集都不能保證沒有發(fā)生變化。
    很難做到根據數據對象的改動精確判斷哪些Query Key對應的結果集受到影響。最簡單的實現方法,就是清空所有SQL包含table1的Query Key。

    (2) Query緩存中,Query Key對應的是數據對象列表,假如不同的Query Key對應的數據對象列表有交集,那么,交集部分的數據對象就是重復存儲的。
    比如,Query Key 1對應的數據對象列表為{a(id = 1), b(id = 2)},Query Key 2對應的數據對象列表為{a(id = 1), c(id = 3)},這個a就在兩個List同時存在了兩份。

    4.二級緩存和Query緩存同步的困惑
    假如,Query緩存中,一個Query Key對應的結果列表為{a (id = 1) , b (id = 2), c (id = 3)}; 二級緩存里面有也id = 1對應的數據對象a。
    這兩個數據對象a之間是什么關系?能夠保持狀態(tài)同步嗎?
    我閱讀Hibernate的相關源碼,沒有發(fā)現兩個緩存之間的這種同步關系。
    或者兩者之間毫無關系。就像我上面所說的,只要表數據發(fā)生變化,相關的Query Key都要被清空。所以不用考慮同步問題?

    四、Lightor的緩存策略
    Lightor是我做的Java開源持久層框架。Lightor的意思是,Lightweight O/R。Hibernate,JDO,EJB CMP這些持久層框架,都是Layer。Lightor算不上Layer,而只是一個Helper。這里的O/R意思不是Object/Relational,而是Object/ResultSet的意思。:-)
    Lightor的緩存策略,主要參照Hibernate的緩存思路,Lightor的緩存也分為 Query緩存和ID緩存。但其中有一點不同,兩者之間并不是毫無聯系的,而是相互關聯的。
    1.Lightor的緩存策略的過程描述:
    (1) 條件查詢的請求一般都包括如下信息:SQL, 對應SQL的參數,起始記錄位置(rowStart),最大記錄個數(maxRows),等。
    (2) Lightor首先根據這些信息組成一個Query Key,根據這個Query Key到Query緩存中查找對應的結果ID列表。注意,這里獲取的是ID列表。
    如果結果ID列表存在于Query緩存,那么根據這個ID列表的每個ID,到ID緩存中取對應的數據對象。如果所有ID對應的數據對象都找到,那個返回這個數據對象結果列表。注意,這里獲取的是整個數據對象(所有字段)的列表。
    如果結果ID列表不存在于Query緩存,或者結果ID列表中的某一個ID不存在于ID緩存,那么,就查詢數據庫,獲取結果列表。然后,把獲取的每個數據對象按照ID放入到ID緩存;并組裝成一個ID列表,按照Query Key存放到Query緩存中。注意,這里是把ID列表,而不是整個對象列表,放入到Query緩存中。
    (3) ID查詢的時候,Lightor先從ID緩存中查找該ID,如果不存在,那么查詢數據庫,把結果放入ID緩存。
    (4) Query Key中的SQL涉及到一些表名,如果這些表的任何數據發(fā)生修改、刪除、增加等操作,這些相關的Query Key都要從緩存中清空。
    2.Lightor的緩存策略的優(yōu)點
    (1) Lightor的ID緩存具有Jive緩存,和Hibernate二級ID緩存的優(yōu)點。ID查詢的時候,如果該ID已經存在于緩存中,那么可以直接取出。節(jié)省了一條數據庫查詢。
    (2) Lightor的Query緩存具有Hibernate的Query緩存的優(yōu)點。條件查詢的時候,如果Query Key已經存在于緩存,那么不需要再查詢數據庫。命中的情況下,一次數據庫查詢也不需要。
    (3) Lightor的Query緩存中,Query Key對應的是ID列表,而不是數據對象列表,真正的數據對象只存在于ID緩存中。所以,不同的Query Key對應的ID列表如果有交集,ID對應的數據對象也不會在ID緩存中重復存儲。
    (4) Lightor的緩存也沒有Jive緩存的最壞情況n + 1次數據庫查詢缺點。
    3.Lightor的緩存策略的缺點
    (1) Lightor的Query緩存具有Hibernate的Query緩存的缺點。條件查詢涉及到的表中,如果有任何一條記錄增加、刪除、或改變,那么緩存中所有和該表相關的Query Key都會失效。
    (2) Lightor的ID緩存也具有hibernate的二級ID緩存具有的缺點。條件查詢的時候,即使ID已經存在于緩存中,也需要重新把數據對象整個從數據庫取出,放入到緩存中。

    五、Query Key的效率
    Query緩存的Query Key的空間和時間開銷比較大。
    Query Key里面存放的東西不少,SQL, 參數,范圍(起始,個數)。
    這里面最大的東西就是SQL。又占地方,又花時間(hashCode, equals)。
    Query Key最關鍵的兩個方法是hashCode和equals,重點是SQL的hashCode和equals。

    Lightor的做法是,由于Lightor直接使用SQL,不用HQL、OQL之類,所以推薦盡量使用static final String的SQL,能夠節(jié)省空間和時間,以至于Query Key的效率能夠相當于ID Key的效率。
    至于Hibernate的QueryKey,有興趣的讀者可以去下載閱讀Hibernate的各個版本的源代碼,跟蹤一下QueryKey的實現優(yōu)化過程。

    六、總結
    這里列一個表,綜合表示Jive, Hibernate, Lightor的緩存策略的特征。
    N + 1問題 重復ID緩存問題 Query緩存支持
    Jive緩存 有 無 不支持
    Hibernate緩存 無 有 支持
    Lightor緩存 無 有 支持

    注:
    “重復ID緩存問題”的含義是,每次條件查詢,不是只取ID列表,而是取出完整對象(所有字段)的列表。這樣,同一個ID對應的數據對象,即使在緩存中已經存在,也可能被重新放入緩存。參見相關緩存的缺點描述。
    “重復ID緩存問題”的負面效應到底有多大,就看你的select id from …(只選擇ID)比你的 select * from … (選擇所有字段)快多少。主要影響因素是,字段的個數,字段值的長度,與數據庫服務器之間網絡傳輸速度。
    不管怎么說,即使選擇所有字段,也只是一次數據庫查詢。而N + 1問題帶來的可能最壞的負面效應(N + 1次數據查詢)卻是非常大的。
    選擇緩存策略的時候,應根據這些情況發(fā)生的概率和正負面效應進行取舍。

    ----- added later

    看到Robbin在04年6月的一篇相關文章。

    Hibernate Iterator JCS分析
    http://www.hibernate.org.cn/71.html

    Hibernate Iterator JCS分析 寫道

    而Hibernate List方式是JDBC的簡單封裝,一次sql就把所有的數據都取出來了,它不會像Iterator那樣先取主鍵,然后再取數據,因此List無法利用JCS。不過List也可以把從數據庫中取出的數據填充到JCS里面去。

    最佳的方式:第一次訪問使用List,快速填充JCS,以后訪問采用Iterator,充分利用JCS。

    posted @ 2006-10-11 08:35 soufan 閱讀(206) | 評論 (0)編輯 收藏

    原文:http://blog.csdn.net/chenlaoshi/archive/2006/09/12/1210564.aspx

    主要就我所了解的J2EE開發(fā)的框架或開源項目做個介紹,可以根據需求選用適當的開源組件進行開發(fā).主要還是以Spring為核心,也總結了一些以前web開發(fā)常用的開源工具和開源類庫
    ?
    1持久層:
    1)Hibernate
    這個不用介紹了,用的很頻繁,用的比較多的是映射,包括繼承映射和父子表映射
    對 于DAO在這里介紹個在它基礎上開發(fā)的包bba96,目前最新版本是bba96 2.0它對Hibernate進行了封裝, 查詢功能包括執(zhí)行hsql或者sql查詢/更新的方法,如果你要多層次邏輯的條件查詢可以自己組裝QueryObject.可以參考它做 HibernateDAO.也可以直接利用它
    2) iBATIS
    另一個ORM工具,Apache的,沒有Hibernate那么集成,自由度比較大
    2:SpringMVC
    ?????? 原理說明和快速入門:
    ?????? 配置文件為:
    Spring的配置文件默認為WEB-INF/xxxx-servelet.xm其中xxx為web.xml中org.springframework.web.servlet.DispatcherServlet的servlet-name。
    ?????? Action分發(fā):
    Spring將按照配置文件定義的URL,Mapping到具體Controller類,再根據URL里的action= xxx或其他參數,利用反射調用Controller里對應的Action方法。
    輸入數據綁定:
    Spring提供Binder 通過名字的一一對應反射綁定Pojo,也可以直接從request.getParameter()取數據。
    輸入數據驗證
    Sping 提供了Validator接口當然還可以使用開源的Commons-Validaor支持最好
    Interceptor(攔截器)
    Spring的攔截器提供接口需要自己編寫,在這點不如WebWork做的好.全面
    ?????? (這里提一下WebWork和Struts的區(qū)別最主要的區(qū)別在于WebWork在建立一個Action時是新New一個對象而Struts是SingleMoule所有的都繼承它的一個Action,所以根據項目需要合適的選擇.)
    3:View層
    1) 標簽庫:JSP2.0/JSTL
    由于Webwork或Spring的標簽確實很有限,一般view層用JSTL標簽,而且據說JSTL設計很好速度是所有標簽中最快的使用起來也很簡單
    ?
    2) 富客戶端:DOJO Widgets, YUI(YahooUI),FCKEditor, Coolest日歷控件
    Dojo主要提供Tree, Tab等富客戶端控件,可以用其進行輔助客戶端開發(fā)
    YahooUI和DOJO一樣它有自己的一套javascript調試控制臺,主要支持ajax開發(fā)也有很多Tree,Table,Menu等富客戶端控件
    FCKEditor 最流行的文本編輯器
    Coolest日歷控件 目前很多日歷控件可用,集成在項目中也比較簡單,這個只是其中的一個,界面不錯的說..
    ?
    3) JavaScript:Prototype.js
    Prototype.js 作為javascript的成功的開源框架,封裝了很多好用的功能,通過它很容易編寫AJAX應用,現在AJAX技術逐漸成熟,框架資源比較豐富,比如 YUI,DWR等等,也是因為JavaScript沒有合適的調試工具,所以沒有必要從零開始編寫AJAX應用,個人認為多用一些成熟的Ajax框架實現 無刷新更新頁面是不錯的選擇.
    ?
    4)表格控件:Display Tag ,Extreme Table
    這兩個的功能差不多,都是View層表格的生成,界面也比較相向,可以導出Excel,Pdf,對Spring支持很容易.
    相比較而言比較推薦ExtremeTable,它的設計很好功能上比DisplayTag多一些,支持Ajax,封裝了一些攔截器,而且最方面的是在主頁wiki中有詳細的中文使用文檔.
    ?
    5):OSCache
    OSCache是OpenSymphony組織提供的一個J2EE架構中Web應用層的緩存技術實現組件,Cache是一種用于提高系統響應速度、改善系統運行性能的技術。尤其是在Web應用中,通過緩存頁面的輸出結果,可以很顯著的改善系統的穩(wěn)定性和運行性能。
    它主要用在處理短時間或一定時間內一些數據或頁面不會發(fā)生變化,或將一些不變的統計報表,緩沖在內存,可以充分的減輕服務器的壓力,防治負載平衡,快速重啟服務器(通過硬盤緩存).
    ?
    6)SiteMesh
    sitemesh 應用Decorator模式主要用于提高頁面的可維護性和復用性,其原理是用Filter截取request和response,把頁面組件head, content,banner結合為一個完整的視圖。通常我們都是用include標簽在每個jsp頁面中來不斷的包含各種header, stylesheet, scripts and footer,現在,在sitemesh的幫助下,我們刪掉他們輕松達到復合視圖模式.
    Sitemesh也是 OpenSymphony的一個項目現在最近的版本是2.2,目前OpenSymphony自從04年就沒有更新的版本了..感覺它還是比較有創(chuàng)新的一種頁面組裝方式, OpenSymphony開源組織的代碼一般寫的比較漂亮,可以改其源代碼對自己的項目進行適配.
    測試發(fā)現Sitemesh還存在一些問題,比如中文問題,它的默認編碼是iso-8859-1在使用時候需要做一些改動.
    ?
    7)CSS,XHTML
    這個不用說了,遵循W3C標準的web頁面開發(fā).
    ?
    8)分頁標簽: pager-taglib組件
    Pager-taglib?是一套分頁標簽庫,可以靈活地實現多種不同風格的分頁導航頁面,并且可以很好的與服務器分頁邏輯分離.使用起來也比較簡單.
    ?
    9)Form: Jodd Form taglib
    Jodd Form taglib使用比較簡單,只要把<form>的頭尾以<jodd:form bean= "mybean">包住
    就會自動綁定mybean, 自動綁定mybean的所有同名屬性到普通html標記input, selectbox, checkbox,radiobox.....在這些input框里不用再寫任何代碼…
    ??????
    10)Ajax:DWR
    ?????? J2EE應用最常用的ajax框架
    ??????
    ?????? 11)報表 圖表
    Eclipse BIRT功能比較強大,也很龐大..好幾十M,一般沒有特別需求或別的圖表設計軟件可以解決的不用它
    JasperReports+ iReport是一個基于Java的開源報表工具,它可以在Java環(huán)境下像其它IDE報表工具一樣來制作報表。JasperReports支持PDF、 HTML、XLS、CSV和XML文件輸出格式。JasperReports是當前Java開發(fā)者最常用的報表工具。
    JFreeChart主要是用來制作各種各樣的圖表,這些圖表包括:餅圖、柱狀圖(普通柱狀圖以及堆棧柱狀圖)、線圖、區(qū)域圖、分布圖、混合圖、甘特圖以及一些儀表盤等等。
    ??????琴棋報表,國產的..重點推薦,適合中國的情況,開放源代碼,使用完全免費。純JAVA開發(fā),適用多種系統平臺。特別適合B/S結構的系統。官方網站有其優(yōu)點介紹,看來用它還是不錯的選擇,最重要的是支持國產呵呵
    ?
    4:權限控制: Acegi
    Acegi是Spring Framework 下最成熟的安全系統,它提供了強大靈活的企業(yè)級安全服務,如完善的認證和授權機制,Http資源訪問控制,Method 調用訪問控制等等,支持CAS
    (耶魯大學的單點登陸技術,這個單點登陸方案比較出名.我也進行過配置使用,可以根據項目需要,如果用戶分布在不同的地方不同的系統通用一套登陸口令可以用它進行解決,一般注冊機登陸機就是這樣解決的)
    ?????? Acegi只是于Spring結合最好的安全框架,功能比較強大,當然還有一些其他的安全框架,這里列舉一些比較流行的是我從網上找到的,使用方法看其官方文檔把…
    JAAS, Seraph, jSai - Servlet Security, Gabriel, JOSSO, Kasai, jPAM, OpenSAML都是些安全控制的框架..真夠多的呵呵
    ?
    5:全文檢索
    ?????? 1) Lucene
    ?????? Lucene是 一套全文索引接口,可以通過它將數據進行倒排文件處理加入索引文件,它的索引速度和查詢速度是相當快的,查詢百萬級數據毫秒級出結果,現在最火的 Apache開源項目,版本更新速度很快現在已經到了2.0,每個版本更新的都比較大,目前用的最多的版本應該是1.4.3,但它有個不太方面的地方單個 索引文件有2G文件限制,現在2.0版本沒有這個限制,我研究的比較多,它的擴展性比較好,可以很方面的擴充其分詞接口和查詢接口.
    ?????? 基于它的開發(fā)的系統很多,比如最常用的Eclipse的搜索功能,還有一些開源的軟件比如Compass,Nutch,Lius,還有我最近做的InSearch(企業(yè)級FTP文件網頁搜索)
    6:公共Util類
    ?????? 主要是Jakarta-Commons類庫,其中最常用得是以下幾個類庫
    1) Jakarta-Commons-Language
    ?????? 最常用得類是StringUtils類,提供了使用的字符串處理的常用方法效率比較高
    2) Jakarta-Commons-Beantuils
    ?????? 主要用Beantuils能夠獲得反射函數封裝及對嵌套屬性,map,array型屬性的讀取。
    3) Jakarta-Commons-Collections
    ?????? 里面有很多Utils方法
    ?
    7 日志管理
    ?????? Log4J
    ?????? 任務是日志記錄,分為Info,Warn,error幾個層次可以更好的調試程序
    ?
    8 開源的J2EE框架
    ?????? 1) Appfuse
    ????????????? Appfuse是Matt Raible 開發(fā)的一個指導性的入門級J2EE框架, 它對如何集成流行的Spring、Hibernate、iBatis、Struts、Xdcolet、JUnit等基礎框架給出了示范. 在持久層,AppFuse采用了Hibernate O/R映射工具;在容器方面,它采用了Spring,用戶可以自由選擇Struts、Spring/MVC,Webwork,JSF這幾個Web框架。
    ??????
    ?????? 2) SpringSide
    ?????? .SpringSide較完整的演示了企業(yè)應用的各個方面,是一個電子商務網站的應用 SpringSide也大量參考了Appfuse中的優(yōu)秀經驗。最重要的是它是國內的一個開源項目,可以了解到國內現在的一些實際技術動態(tài)和方向很有指導意義…
    ?
    9:模版 Template
    主要有Veloctiy和Freemarker
    模板用Servlet提供的數據動態(tài)地生成 HTML。編譯器速度快,輸出接近靜態(tài)HTML???????????? 頁面的速度。
    ?
    10:工作流
    ?????? 我所知道比較出名的主要有JBpm Shark Osworkflow,由于對它沒有過多的研究所以還不是很清楚之間有什么區(qū)別.
    ?
    項目管理軟件
    dotProject:是一個基于LAMP的開源項目管理軟件。最出名的項目管理軟件
    JIRA: 項目計劃,任務安排,錯誤管理
    Bugzilla:提交和管理bug,和eclipse集成,可以通過安裝MyEclipse配置一下即可使用
    BugFree借鑒微軟公司軟件研發(fā)理念、免費開放源代碼、基于Web的精簡版Bug管理
    CVS:這個就不介紹了都在用.
    SVN: SubVersion已逐漸超越CVS,更適應于JavaEE的項目。Apache用了它很久后,Sourceforge剛剛推出SVN的支持。
    測試用例:主要JUnit單元測試,編寫TestCase,Spring也對Junit做了很好的支持
    ?
    后記:
    ?????? 以Spring 為主的應用開發(fā)可選用的組件中間件真是眼花繚亂,所以針對不同的項目需求可以利用不同的開源產品解決,比如用Spring+Hibernate/ iBATIS或Spring+WebWork+Hibernate/ iBATIS或Spring+Struts+Hibernate/ iBATIS,合理的框架設計和代碼復用設計對項目開發(fā)效率和程序性能有很大的提高,也有利于后期的維護.
    posted @ 2006-09-28 20:57 soufan 閱讀(102) | 評論 (0)編輯 收藏

    Hashtable和HashMap的區(qū)別:
    1.Hashtable是Dictionary的子類,HashMap是Map接口的一個實現類;
    2.Hashtable中的方法是同步的,而HashMap中的方法在缺省情況下是非同步的。即是說,在多線程應用程序中,不用專門的操作就安全地可以使用Hashtable了;而對于HashMap,則需要額外的同步機制。但HashMap的同步問題可通過Collections的一個靜態(tài)方法得到解決:
    Map Collections.synchronizedMap(Map m)
    這個方法返回一個同步的Map,這個Map封裝了底層的HashMap的所有方法,使得底層的HashMap即使是在多線程的環(huán)境中也是安全的。
    3.在HashMap中,null可以作為鍵,這樣的鍵只有一個;可以有一個或多個鍵所對應的值為null。當get()方法返回null值時,即可以表示HashMap中沒有該鍵,也可以表示該鍵所對應的值為null。因此,在HashMap中不能由get()方法來判斷HashMap中是否存在某個鍵,而應該用containsKey()方法來判斷。

    Vector、ArrayList和List的異同

    線性表,鏈表,哈希表是常用的數據結構,在進行Java開發(fā)時,JDK已經為我們提供了一系列相應的類來實現基本的數據結構。這些類均在java.util包中。本文試圖通過簡單的描述,向讀者闡述各個類的作用以及如何正確使用這些類。

    Collection
    ├List
    │├LinkedList
    │├ArrayList
    │└Vector
    │ └Stack
    └Set
    Map
    ├Hashtable
    ├HashMap
    └WeakHashMap

    Collection接口
      Collection是最基本的集合接口,一個Collection代表一組Object,即Collection的元素(Elements)。一些Collection允許相同的元素而另一些不行。一些能排序而另一些不行。Java SDK不提供直接繼承自Collection的類,Java SDK提供的類都是繼承自Collection的“子接口”如List和Set。
      所有實現Collection接口的類都必須提供兩個標準的構造函數:無參數的構造函數用于創(chuàng)建一個空的Collection,有一個Collection參數的構造函數用于創(chuàng)建一個新的Collection,這個新的Collection與傳入的Collection有相同的元素。后一個構造函數允許用戶復制一個Collection。
      如何遍歷Collection中的每一個元素?不論Collection的實際類型如何,它都支持一個iterator()的方法,該方法返回一個迭代子,使用該迭代子即可逐一訪問Collection中每一個元素。典型的用法如下:
        Iterator it = collection.iterator(); // 獲得一個迭代子
        while(it.hasNext()) {
          Object obj = it.next(); // 得到下一個元素
        }
      由Collection接口派生的兩個接口是List和Set。

    List接口
      List是有序的Collection,使用此接口能夠精確的控制每個元素插入的位置。用戶能夠使用索引(元素在List中的位置,類似于數組下標)來訪問List中的元素,這類似于Java的數組。
    和下面要提到的Set不同,List允許有相同的元素。
      除了具有Collection接口必備的iterator()方法外,List還提供一個listIterator()方法,返回一個ListIterator接口,和標準的Iterator接口相比,ListIterator多了一些add()之類的方法,允許添加,刪除,設定元素,還能向前或向后遍歷。
      實現List接口的常用類有LinkedList,ArrayList,Vector和Stack。

    LinkedList類
      LinkedList實現了List接口,允許null元素。此外LinkedList提供額外的get,remove,insert方法在LinkedList的首部或尾部。這些操作使LinkedList可被用作堆棧(stack),隊列(queue)或雙向隊列(deque)。
      注意LinkedList沒有同步方法。如果多個線程同時訪問一個List,則必須自己實現訪問同步。一種解決方法是在創(chuàng)建List時構造一個同步的List:
        List list = Collections.synchronizedList(new LinkedList(...));

    ArrayList類
      ArrayList實現了可變大小的數組。它允許所有元素,包括null。ArrayList沒有同步。
    size,isEmpty,get,set方法運行時間為常數。但是add方法開銷為分攤的常數,添加n個元素需要O(n)的時間。其他的方法運行時間為線性。
      每個ArrayList實例都有一個容量(Capacity),即用于存儲元素的數組的大小。這個容量可隨著不斷添加新元素而自動增加,但是增長算法并沒有定義。當需要插入大量元素時,在插入前可以調用ensureCapacity方法來增加ArrayList的容量以提高插入效率。
      和LinkedList一樣,ArrayList也是非同步的(unsynchronized)。

    Vector類
      Vector非常類似ArrayList,但是Vector是同步的。由Vector創(chuàng)建的Iterator,雖然和ArrayList創(chuàng)建的Iterator是同一接口,但是,因為Vector是同步的,當一個Iterator被創(chuàng)建而且正在被使用,另一個線程改變了Vector的狀態(tài)(例如,添加或刪除了一些元素),這時調用Iterator的方法時將拋出ConcurrentModificationException,因此必須捕獲該異常。

    Stack 類
      Stack繼承自Vector,實現一個后進先出的堆棧。Stack提供5個額外的方法使得Vector得以被當作堆棧使用。基本的push和pop方法,還有peek方法得到棧頂的元素,empty方法測試堆棧是否為空,search方法檢測一個元素在堆棧中的位置。Stack剛創(chuàng)建后是空棧。

    Set接口
      Set是一種不包含重復的元素的Collection,即任意的兩個元素e1和e2都有e1.equals(e2)=false,Set最多有一個null元素。
      很明顯,Set的構造函數有一個約束條件,傳入的Collection參數不能包含重復的元素。
      請注意:必須小心操作可變對象(Mutable Object)。如果一個Set中的可變元素改變了自身狀態(tài)導致Object.equals(Object)=true將導致一些問題。

    Map接口
      請注意,Map沒有繼承Collection接口,Map提供key到value的映射。一個Map中不能包含相同的key,每個key只能映射一個value。Map接口提供3種集合的視圖,Map的內容可以被當作一組key集合,一組value集合,或者一組key-value映射。

    Hashtable類
      Hashtable繼承Map接口,實現一個key-value映射的哈希表。任何非空(non-null)的對象都可作為key或者value。
      添加數據使用put(key, value),取出數據使用get(key),這兩個基本操作的時間開銷為常數。
    Hashtable通過initial capacity和load factor兩個參數調整性能。通常缺省的load factor 0.75較好地實現了時間和空間的均衡。增大load factor可以節(jié)省空間但相應的查找時間將增大,這會影響像get和put這樣的操作。
    使用Hashtable的簡單示例如下,將1,2,3放到Hashtable中,他們的key分別是”one”,”two”,”three”:
        Hashtable numbers = new Hashtable();
        numbers.put(“one”, new Integer(1));
        numbers.put(“two”, new Integer(2));
        numbers.put(“three”, new Integer(3));
      要取出一個數,比如2,用相應的key:
        Integer n = (Integer)numbers.get(“two”);
        System.out.println(“two = ” + n);
      由于作為key的對象將通過計算其散列函數來確定與之對應的value的位置,因此任何作為key的對象都必須實現hashCode和equals方法。hashCode和equals方法繼承自根類Object,如果你用自定義的類當作key的話,要相當小心,按照散列函數的定義,如果兩個對象相同,即obj1.equals(obj2)=true,則它們的hashCode必須相同,但如果兩個對象不同,則它們的hashCode不一定不同,如果兩個不同對象的hashCode相同,這種現象稱為沖突,沖突會導致操作哈希表的時間開銷增大,所以盡量定義好的hashCode()方法,能加快哈希表的操作。
      如果相同的對象有不同的hashCode,對哈希表的操作會出現意想不到的結果(期待的get方法返回null),要避免這種問題,只需要牢記一條:要同時復寫equals方法和hashCode方法,而不要只寫其中一個。
      Hashtable是同步的。

    HashMap類
      HashMap和Hashtable類似,不同之處在于HashMap是非同步的,并且允許null,即null value和null key。,但是將HashMap視為Collection時(values()方法可返回Collection),其迭代子操作時間開銷和HashMap的容量成比例。因此,如果迭代操作的性能相當重要的話,不要將HashMap的初始化容量設得過高,或者load factor過低。

    WeakHashMap類
      WeakHashMap是一種改進的HashMap,它對key實行“弱引用”,如果一個key不再被外部所引用,那么該key可以被GC回收。

    總結
      如果涉及到堆棧,隊列等操作,應該考慮用List,對于需要快速插入,刪除元素,應該使用LinkedList,如果需要快速隨機訪問元素,應該使用ArrayList。
      如果程序在單線程環(huán)境中,或者訪問僅僅在一個線程中進行,考慮非同步的類,其效率較高,如果多個線程可能同時操作一個類,應該使用同步的類。
      要特別注意對哈希表的操作,作為key的對象要正確復寫equals和hashCode方法。
      盡量返回接口而非實際的類型,如返回List而非ArrayList,這樣如果以后需要將ArrayList換成LinkedList時,客戶端代碼不用改變。這就是針對抽象編程。

    posted @ 2006-09-08 17:45 soufan 閱讀(3410) | 評論 (0)編輯 收藏

         摘要: (轉載文章) 1 什么是Java、Java2、JDK?JDK后面的1.3、1.4.2版本號又是怎么回事?   答:Java是一種通用的,并發(fā)的,強類型的,面向對象的編程語言(摘自Java規(guī)范第二版) JDK是Sun公司分發(fā)的免費Java開發(fā)工具,正式名稱為J2SDK(Java2 Software Develop Kit)。 ...  閱讀全文
    posted @ 2006-09-08 17:43 soufan 閱讀(228) | 評論 (0)編輯 收藏

    摘自:ChinaITLab 作者: 瀏覽率:70

      JSF對通過關聯組件和事件來構建頁面而說是非常棒的,但是,與所有現有的技術一樣,它需要一個控制器來分離出頁面間的導航決策,并提供到業(yè)務層的鏈接。它擁有一個基本的導航處理程序,可以用功能完備的處理程序來替換它。Page Flow為創(chuàng)建可重用的封裝頁面流提供了基礎,并可以與視圖層并行工作。它是一個功能完備的導航處理程序,將JSF頁面作為最優(yōu)先的處理對象。本文將討論如何集成這兩種技術來利用二者的優(yōu)點。

      構建Beehive/JSF應用程序

      要構建Beehive/JSF應用程序,首先要啟動Page Flow,然后添加對JSF的支持。起點是從基本的支持NetUI(Beehive中包含Page Flow的組件)的項目開始。根據指導構建基本的支持NetUI的Web應用程序。在本文中,我們暫且稱之為“jsf-beehive”,可以在 http://localhost:8080/jsf-beehive 上獲得。

      接下來,安裝并配置JSF。Page Flow可以使用任何與JavaServer Faces 1.1兼容的實現,并針對兩種主流實現進行了測試:Apache MyFaces和JSF Reference Implementation。根據下面的指導在新的Web應用程序中安裝JSF:MyFaces v1.0.9及更高版本,JSF Reference Implementation v1.1_01,或者其他實現。之后,可以使用WEB-INF/faces-config.xml中的一個簡單入口啟動Page Flow集成,入口在<application>標簽之下,<navigation-rule>標簽之上:

    																		<factory>
     <application-factory>
      org.apache.beehive.netui.pageflow.faces.PageFlowApplicationFactory
     </application-factory>
    </factory>
    																

      添加了這些就為頁面流提供了一個機會,使其可以提供自己的JSF框架對象版本來定制其行為。通常來說,只有在使用頁面流功能的時候,JSF行為才會被修改;JSF的基本行為不會改變。

      基本集成

      JSF中頁面流的最基本用處是引發(fā)(調用)來自JSF頁面的動作。JSF頁面可以處理頁面內事件,而頁面流動作則是從一個頁面導航到另一頁面的方法。首先,在Web應用程序中創(chuàng)建一個名為“example”的目錄,在其中創(chuàng)建一個頁面流控制器類:

    																		package example;
    
    import org.apache.beehive.netui.pageflow.Forward;
    import org.apache.beehive.netui.pageflow.PageFlowController;
    import org.apache.beehive.netui.pageflow.annotations.Jpf;
    
    @Jpf.Controller(
      simpleActions={
        @Jpf.SimpleAction(name="begin", path="page1.faces")
      }
    )
    public class ExampleController extends PageFlowController
    {
      @Jpf.Action(
        forwards={
          @Jpf.Forward(name="success", path="page2.faces")
        }
      )
      public Forward goPage2()
      {
        Forward fwd = new Forward("success");
        return fwd;
      }
    }
    
    																

      在這個頁面流中有兩個動作:跳轉到page1.faces的begin動作和跳轉到page2.faces的goPage2動作。將goPage2作為一個方法動作(而不是簡單動作)的原因是稍后將會對其進行擴充。

      在構造頁面的時候,應當以.jsp為擴展名創(chuàng)建page1和page2;JSF servlet處理每個.faces請求,并最終跳轉到相關的JSP。所以,跳轉到page1.faces最終將顯示page1.jsp,如下:

    																		<%@ taglib prefix="f" uri="http://java.sun.com/jsf/core" %>
    <%@ taglib prefix="h" uri="http://java.sun.com/jsf/html" %>
    ?
    <html>
     <body>
       <f:view>
         <h:form>
           <h:panelGrid>
             <h:outputText value="Page 1 of page flow #{pageFlow.URI}"/>
             <h:commandLink action="goPage2" value="Go to page 2"/>
           </h:panelGrid>
         </h:form>
       </f:view>
     </body>
    </html>
    																

      從JSF頁面引發(fā)一個動作很簡單:使用命令組件的action屬性中的動作名字就可以了。在上面的例子中,commandLink指向goPage2動作。使用頁面流集成,這意味著goPage2動作會在example.ExampleController中運行。

      就是這樣。要試驗的話,構建應用程序,點擊 http://localhost:8080/jsf-beehive/example/ExampleController.jpf ,這將通過begin動作跳轉到page1.faces。單擊鏈接“Go to page 2”,會引發(fā)goPage2動作并跳轉到page2.faces。

      后臺Bean

      Page Flow框架可以管理與JSF頁面相關的后臺bean(backing bean)。該類是放置與頁面相關的事件處理程序和狀態(tài)的方便場所。可以把它看作是集中放置與頁面交互時所運行的所有代碼的單一場所。當點擊一個JSF頁面時,Page Flow會判斷是否有具有同樣名稱和包的類,例如,page /example/page1.faces的example.page1類。如果存在這樣的類,并且它用@Jpf.FacesBacking進行注釋并擴展了FacesBackingBean,它就會創(chuàng)建該類的一個實例。當離開JSF頁面而轉到一個動作或者其它任何頁面時,后臺bean會被銷毀。后臺bean與JSF頁面共存亡。

      綁定到后臺bean中的屬性

      下面是page1.faces的一個非常簡單的后臺bean,以及屬性someProperty。文件名是page1.java:

    																		package example;
    
    import org.apache.beehive.netui.pageflow.FacesBackingBean;
    import org.apache.beehive.netui.pageflow.annotations.Jpf;
    
    @Jpf.FacesBacking
    public class page1 extends FacesBackingBean
    {
      private String _someProperty = "This is a property value from" 
                                     + getClass().getName() + ".";
    
      public String getSomeProperty()
      {
          return _someProperty;
      }
    
      public void setSomeProperty(String someProperty)
      {
          _someProperty = someProperty;
      }
    }
    
    																

      在JSF頁面(page1.jsp)中,可以利用backing綁定上下文來綁定到這個屬性:

      <h:outputText value="#{backing.someProperty}"/>

      上面的例子顯示了someProperty(最終在后臺bean上調用getSomeProperty())的值。類似地,設置這個值:

      <h:inputText value="#{backing.someProperty}"/>

      注意,在這個例子中,后臺bean中沒有出現事件處理程序或組件引用。這就縮短了代碼;后臺bean是放置頁面所有的處理程序和組件引用的好地方。

      從后臺bean引發(fā)頁面流動作

      在上面的“基本集成”部分,我們直接從JSF組件引發(fā)頁面流動作。通常情況下,只需這樣即可;當單擊一個按鈕或者鏈接時,會運行一個動作并跳轉到另一個頁面上。如果想在調用控制器之前運行一些與頁面相關的代碼,或者如果希望頁面可以在幾個動作之間進行動態(tài)選擇的話,可以在命令處理程序(JSF頁面所運行的一個Java方法)中引發(fā)一個動作。下面是一個命令處理程序的例子,可以把它放到后臺bean page2.java中(或者其它任何可公開訪問的bean中):

    																		public String
    chooseNextPage()
    {
      return "goPage3";
    }
    																

      這是一個非常簡單的命令處理程序,它選擇了goPage3動作。可以用標準的JSF方式從一個JSF命令組件綁定到這個命令處理程序:

    																		<h:commandButton action="#{backing.chooseNextPage}" 
                     value="Submit"/>
    																

      當單擊鏈接時,會運行chooseNextPage命令處理程序,它會選擇引發(fā)goPage3動作。還可以對命令處理程序方法使用一個特殊的頁面流注釋——@Jpf.CommandHandler:

    																		@Jpf.CommandHandler(
     raiseActions={
          @Jpf.RaiseAction(action="goPage3")
     }
    )
    public String chooseNextPage()
    {
     return "goPage3";
    }
    																

      該注釋使支持Beehive的工具可以知道命令處理程序引發(fā)了后臺bean中的哪個動作,并允許擴展JSF動作處理的能力(參見下面“從JSF頁面向頁面流發(fā)送數據”部分)。

      從后臺bean訪問當前頁面流或共享流

      在某些情況下,您或許想直接從后臺bean訪問當前頁面流或一個活動的共享流。為此,只需創(chuàng)建一個適當類型的字段,并使用@Jpf.PageFlowField或@Jpf.SharedFlowField對其進行適當注釋:

    																		@Jpf.CommandHandler(
     raiseActions={
          @Jpf.RaiseAction(action="goPage3")
     }
    )
    public String chooseNextPage()
    {
     return "goPage3";
    }
    
    																

      這些字段將在創(chuàng)建后臺bean的時候被初始化。無需手動對其進行初始化。下面的例子使用了自動初始化的ExampleController字段。在這個例子中,“show hints”單選鈕的事件處理程序在頁面流中設置了一個普通優(yōu)先級。

    																		@Jpf.PageFlowField
    private ExampleController myController;
    
    @Jpf.SharedFlowField(name="sharedFlow2") // "sharedFlow2" is a 
                                  // name defined in the
                                  // page flow controller
    private ExampleSharedFlow mySharedFlow;
    
    
    																

      在很多情況下,頁面不需要直接與頁面流或者共享流進行交互;使用其它方法從頁面流向JSF頁面?zhèn)鬟f數據就足夠了,反之亦然。下面我將給出一些例子。

      從頁面流控制器訪問后臺bean

      您不能從頁面流控制器訪問后臺bean!至少,這不容易做到,這是有意為之的。后臺bean與JSF頁面緊密相關,當您離開頁面的時候,后臺bean會被銷毀。正如頁面流控制器不應了解頁面細節(jié)一樣,它也不應了解后臺bean。當然了,可以從后臺bean向控制器傳遞數據(稍后將會介紹),甚至可以傳遞后臺bean實例本身,但是在大多數情況下,后臺bean的內容是不應當泄露給控制器的。

      生命周期方法

      通常,當后臺bean發(fā)生某些事情的時候,比如當它被創(chuàng)建或銷毀時,我們希望能運行代碼。在Page Flow框架的生命周期中,它會對后臺bean調用一些方法:

    • onCreate():創(chuàng)建bean時
    • onDestroy():銷毀bean時(從用戶會話移除)
    • onRestore():這個需要詳細解釋一下。我說過,當您離開頁面的時候,后臺bean會被銷毀。在大多數情況下是這樣的,但是如果頁面流使用了navigateTo特性(它使您可以再次訪問先前顯示的頁面),在您離開頁面之后,Page Flow框架會保留后臺bean一小段時間,以防它需要還原。當通過@Jpf.Forward或@Jpf.SimpleAction使用navigateTo=Jpf.NavigateTo.currentPage或navigateTo=Jpf.NavigateTo.previousPage還原一個JSF頁面時,頁面的組件樹及其后臺bean都被Page Flow框架還原。當這種情況發(fā)生時,onRestore()就被調用。

      不管要在哪個時期運行代碼,只需重寫適當的方法:

    																		protected void onCreate()
    {
     /*some create-time logic */
    }
    																

      當重寫這些方法時,不需要調用空的super版本。

      在JSF頁面和頁面流之間傳遞數據

      現在我們該看看如何在JSF頁面和頁面流之間傳遞數據了。

      從頁面流向JSF頁面發(fā)送數據

      通常,您會想要利用頁面流的數據來初始化一個頁面。為此,可以向page2.faces的Forward添加“action outputs”:

    																		@Jpf.Action(
     forwards={
      @Jpf.Forward(
        name="success", path="page2.faces",
        actionOutputs={
          @Jpf.ActionOutput(name="message", type=String.class,required=true)
        }
      )
     }
    )
    
    public Forward goPage2()
    {
      Forward fwd = new
      Forward("success");
       fwd.addActionOutput("message", "Got the message.");
      return fwd;
    }
    
    																

      做完這些之后,可以直接從JSF頁面或者后臺bean將該值作為頁面輸入來訪問。(如果您不喜歡鍵入冗長的注釋,可以省去斜體的。它們主要用于再次檢查添加的對象類型是否正確,確定不缺失類型。)

      可以在頁面中利用JSF表示語言中的頁面流pageInput綁定上下文綁定到這個值:

    																		<h:outputText value="#{pageInput.message}"/>
    																

      注意,可以利用pageFlow和sharedFlow綁定上下文綁定到頁面流控制器自身或者任何可用的共享流的屬性:

    																		<h:outputText value="#{pageFlow.someProperty}"/>
    <h:outputText value="#{sharedFlow.mySharedFlow.someProperty}"/>
    																

      最后,要想從后臺bean訪問頁面輸入,只需在bean類代碼中的任意地方調用getPageInput:

    																		String message = (String) getPageInput("message");
    																

      從JSF頁面向頁面流發(fā)送數據

      還可以隨著頁面流所引發(fā)的動作發(fā)送數據。很多動作將要求表單bean作為輸入;通常,表單bean用于從頁面獲取數據送到控制器。首先,讓我們構建一個動作來接收表單bean并跳轉到頁面:

    																		@Jpf.Action(
       forwards={
           @Jpf.Forward(name="success", path="page3.faces")
       }
    )
    public Forward goPage3(NameBean nameBean)
    {
        _userName = nameBean.getFirstName() + ' ' + 
                    nameBean.getLastName();
        return new Forward("success");
    }
    
    																

      該動作包含一個NameBean,它是一個將getters/setters作為其firstName和lastName屬性的表單bean類。它設置一個成員變量保存完整名字,之后跳轉到page3.faces。我們知道,可以直接從JSF頁面或者它的后臺bean引發(fā)一個動作。在這兩種情況下,都可以向動作發(fā)送表單bean。下面讓我們依次看看每種情況。

      從后臺bean發(fā)送表單bean

      要從后臺bean中的命令處理程序發(fā)送表單bean,需要使用一個特定的注釋。下面給出了page2.java中的情況:

    																		private ExampleController.NameBean _nameBean;
    
    protected void onCreate()
    {
        _nameBean = new ExampleController.NameBean();
    }
    
    public ExampleController.NameBean getName()
    {
        return _nameBean;
    }
    
    @Jpf.CommandHandler(
        raiseActions={
            @Jpf.RaiseAction(action="goPage3", 
                 outputFormBean="_nameBean")
        }
    )
    public String chooseNextPage()
    {
        return "goPage3";
    }
    
    																

      在這個例子中,JSF頁面可以用它選擇的任何方式填充_nameBean的值(例如,通過將h:inputText值綁定到#{backing.name.firstName}和#{backing.name.lastName})。之后它使用@Jpf.RaiseAction上的outputFormBean屬性來標記_nameBean應當被傳遞到動作goPage3。

      從JSF頁面發(fā)送表單bean

      從JSF頁面直接發(fā)送表單bean很容易,只要您可以通過數據綁定表達式得到bean值。這是通過在commandButton組件內部添加名為submitFormBean的h:attribute組件來實現的:

    																		<h:commandButton action="#{backing.chooseNextPage}" 
                     value="Submit directly from page">
        <f:attribute name="submitFormBean" value="backing.name" />
    </h:commandButton>
    																

      在這里,為了使表單bean發(fā)送到動作goPage3,按鈕綁定到后臺bean的“name”屬性(getName)。

      結束語

      本文展示了如何將JSF在構建頁面方面的豐富特性與Beehive Page Flow在控制頁面間導航方面的強大功能相結合。二者的集成非常容易,但是卻會對應用造成深遠的影響:它將JSF頁面與應用級邏輯相分離,并把頁面帶入Page Flow所提供的功能領域中。JSF頁面得到了清楚的任務:作為單個(如果有足夠能力的話)視圖元素參與到應用程序的流中。文中沒有展示JSF頁面中具有事件處理功能且控制器中具有復雜的導航邏輯的完備應用程序。但是隨著應用程序的復雜程度提高,它就會更加需要責任的劃分以及頁面流添加給JSF的高級流功能。您可以花幾分鐘嘗試一下——很快您就將意識到這樣做所帶來的好處。

    posted @ 2006-09-03 02:40 soufan 閱讀(205) | 評論 (0)編輯 收藏

    主站蜘蛛池模板: 国产99视频精品免费观看7| 丰满少妇作爱视频免费观看| 亚洲日本香蕉视频| 久久夜色精品国产噜噜亚洲AV| 亚洲日韩小电影在线观看| 亚洲精品国精品久久99热一| 国产精品一区二区三区免费 | 天天摸夜夜摸成人免费视频 | 中文字幕免费在线看电影大全 | jjizz全部免费看片| 99久久久国产精品免费无卡顿| 69天堂人成无码麻豆免费视频| 亚洲人成电影网站免费| 色播在线永久免费视频| 免费国产成人α片| 8x成人永久免费视频| 性xxxxx免费视频播放| 免费羞羞视频网站| 亚洲裸男gv网站| 亚洲动漫精品无码av天堂| 亚洲高清视频在线播放| 亚洲手机中文字幕| 亚洲国产成人AV在线播放 | 亚洲精品A在线观看| 亚洲中文字幕无码日韩| 亚洲日本精品一区二区| 亚洲av无码专区在线| 亚洲国产aⅴ成人精品无吗| 国产福利在线观看永久免费| 国产好大好硬好爽免费不卡 | sss日本免费完整版在线观看| 国产午夜无码精品免费看动漫| 中文字幕免费视频| 午夜时刻免费入口| 久久亚洲国产精品123区| 亚洲理论精品午夜电影| 亚洲av日韩专区在线观看| 岛国岛国免费V片在线观看| 91九色老熟女免费资源站| 国产精品极品美女免费观看| 亚洲国产成人片在线观看无码 |