Java Tools

            BlogJava :: 首頁(yè) :: 新隨筆 :: 聯(lián)系 :: 聚合  :: 管理 ::
            83 隨筆 :: 0 文章 :: 16 評(píng)論 :: 0 Trackbacks

          #

          軟件需求說(shuō)明書(shū)
          1. 引言
          1.1 項(xiàng)目名稱(chēng)
          1.2 項(xiàng)目背景和內(nèi)容概要
          (項(xiàng)目的委托單位、開(kāi)發(fā)單位、主管部門(mén)、與其它項(xiàng)目的關(guān)系,與其他機(jī)構(gòu)的關(guān)系等)

          1.3 相關(guān)資料、縮略語(yǔ)、定義
          (相關(guān)項(xiàng)目計(jì)劃、合同及上級(jí)機(jī)關(guān)批文,引用的文件、采用的標(biāo)準(zhǔn)等)
          (縮寫(xiě)詞和名詞定義)
          2. 任務(wù)概述
          2.1 目標(biāo)
          (項(xiàng)目的開(kāi)發(fā)目標(biāo)和應(yīng)用目標(biāo)。如果是其他系統(tǒng)的一部分,則說(shuō)明其關(guān)系)
          2.2 范圍
          (包含的業(yè)務(wù),不包含的業(yè)務(wù))
          2.3 假定條件與約束限制
          (盡量列出開(kāi)展本項(xiàng)目的假定和約束,例如:經(jīng)費(fèi)限制,開(kāi)發(fā)期限,設(shè)備條件,用戶現(xiàn)
          場(chǎng)環(huán)境準(zhǔn)備等)
          3.業(yè)務(wù)流程
          4.?dāng)?shù)據(jù)描述
          4.1 原始數(shù)據(jù)描述
          a. 靜態(tài)數(shù)據(jù)
          b. 動(dòng)態(tài)數(shù)據(jù)
          4.2 數(shù)據(jù)流向圖
          4.3 數(shù)據(jù)概念模型和描述
          5.功能需求
          5.1 功能描述
          6.界面要求
          6.1報(bào)表格式
          6.2圖形要求
          6.3輸入輸出要求
          7.接口要求
          (描述與本系統(tǒng)相連的系統(tǒng)的接口的數(shù)據(jù)格式,數(shù)據(jù)交換協(xié)議,接口功能等)
          8.性能需求
          8.1數(shù)據(jù)精確度
          (例如,數(shù)據(jù)內(nèi)部精度,外部顯示精度)
          8. 2數(shù)據(jù)量
          8. 3時(shí)間特性要求
          (根據(jù)所開(kāi)發(fā)系統(tǒng)的特點(diǎn),規(guī)定系統(tǒng)對(duì)時(shí)間的特性的要求。例如:
          系統(tǒng)響應(yīng)時(shí)間、界面更新處理時(shí)間、數(shù)據(jù)轉(zhuǎn)換與傳輸時(shí)間)
          9. 運(yùn)行環(huán)境需求
          9.1網(wǎng)絡(luò)和硬件設(shè)備平臺(tái)
          (網(wǎng)絡(luò)拓?fù)鋱D及設(shè)備類(lèi)型描述)
          操作系統(tǒng)平臺(tái)
          數(shù)據(jù)庫(kù)系統(tǒng)平臺(tái)
          10.1編程工具
          10.2其它支撐軟件
          11. 其它專(zhuān)門(mén)需求
          11.1安裝和操作
          11.2安全保密
          11.3維護(hù)服務(wù)
          posted @ 2007-07-02 13:13 和田雨 閱讀(4636) | 評(píng)論 (1)編輯 收藏

          法則1:優(yōu)先使用(對(duì)象)組合,而非(類(lèi))繼承

          [ Favor Composition Over Inheritance ]




          • 組合



            1.(對(duì)象)組合是一種通過(guò)創(chuàng)建一個(gè)組合了其它對(duì)象的對(duì)象,從而獲得新功能的復(fù)用方法。

            2.將功能委托給所組合的一個(gè)對(duì)象,從而獲得新功能。

            3.有些時(shí)候也稱(chēng)之為"聚合"(aggregation)或"包容"(containment),盡管有些作者對(duì)這些術(shù)語(yǔ)賦予了專(zhuān)門(mén)的含義

            4.例如:

            a.聚合:一個(gè)對(duì)象擁有另一個(gè)對(duì)象或?qū)α硪粋€(gè)對(duì)象負(fù)責(zé)(即一個(gè)對(duì)象包含另一個(gè)對(duì)象或是另一個(gè)對(duì)象的一部分),并且聚合對(duì)象和其所有者具有相同的生命周期。(譯者注:即所謂的"同生共死"關(guān)系,可參見(jiàn)GOF的Design
            Patterns: Elements of Reusable Object-Oriented Software的引言部分。)

            b.包容:一種特殊類(lèi)型的組合,對(duì)于其它對(duì)象而言,容器中的被包含對(duì)象是不可見(jiàn)的,其它對(duì)象僅能通過(guò)容器對(duì)象來(lái)訪問(wèn)被包含對(duì)象。(Coad)





            5.包含可以通過(guò)以下兩種方式實(shí)現(xiàn):

            a.根據(jù)引用(By reference)

            b.根據(jù)值(By value)

            6.C++允許根據(jù)值或引用來(lái)實(shí)現(xiàn)包含。

            7.但是在Java中,一切皆為對(duì)象的引用!




          • 組合的優(yōu)點(diǎn)和缺點(diǎn)






            1.優(yōu)點(diǎn):

            a.容器類(lèi)僅能通過(guò)被包含對(duì)象的接口來(lái)對(duì)其進(jìn)行訪問(wèn)。

            b."黑盒"復(fù)用,因?yàn)楸话瑢?duì)象的內(nèi)部細(xì)節(jié)對(duì)外是不可見(jiàn)。

            c.對(duì)裝性好。

            d.實(shí)現(xiàn)上的相互依賴性比較小。(譯者注:被包含對(duì)象與容器對(duì)象之間的依賴關(guān)系比較少)

            e.每一個(gè)類(lèi)只專(zhuān)注于一項(xiàng)任務(wù)。

            f.通過(guò)獲取指向其它的具有相同類(lèi)型的對(duì)象引用,可以在運(yùn)行期間動(dòng)態(tài)地定義(對(duì)象的)組合。




            2.缺點(diǎn):

            a.從而導(dǎo)致系統(tǒng)中的對(duì)象過(guò)多。

            b為了能將多個(gè)不同的對(duì)象作為組合塊(composition block)來(lái)使用,必須仔細(xì)地對(duì)接口進(jìn)行定義。




          • 繼承





            1.(類(lèi))繼承是一種通過(guò)擴(kuò)展一個(gè)已有對(duì)象的實(shí)現(xiàn),從而獲得新功能的復(fù)用方法。

            2.泛化類(lèi)(超類(lèi))可以顯式地捕獲那些公共的屬性和方法。

            3.特殊類(lèi)(子類(lèi))則通過(guò)附加屬性和方法來(lái)進(jìn)行實(shí)現(xiàn)的擴(kuò)展。





          • 繼承的優(yōu)點(diǎn)和缺點(diǎn)





            1.優(yōu)點(diǎn):

            a.容易進(jìn)行新的實(shí)現(xiàn),因?yàn)槠浯蠖鄶?shù)可繼承而來(lái)。

            b.易于修改或擴(kuò)展那些被復(fù)用的實(shí)現(xiàn)。

            2.缺點(diǎn):

            a.破壞了封裝性,因?yàn)檫@會(huì)將父類(lèi)的實(shí)現(xiàn)細(xì)節(jié)暴露給子類(lèi)。

            b. "白盒"復(fù)用,因?yàn)楦割?lèi)的內(nèi)部細(xì)節(jié)對(duì)于子類(lèi)而言通常是可見(jiàn)的。

            c.當(dāng)父類(lèi)的實(shí)現(xiàn)更改時(shí),子類(lèi)也不得不會(huì)隨之更改。

            d.從父類(lèi)繼承來(lái)的實(shí)現(xiàn)將不能在運(yùn)行期間進(jìn)行改變。





          • Coad規(guī)則





            僅當(dāng)下列的所有標(biāo)準(zhǔn)被滿足時(shí),方可使用繼承:

            a.子類(lèi)表達(dá)了"是一個(gè)…的特殊類(lèi)型",而非"是一個(gè)由…所扮演的角色"。

            b子類(lèi)的一個(gè)實(shí)例永遠(yuǎn)不需要轉(zhuǎn)化(transmute)為其它類(lèi)的一個(gè)對(duì)象。

            c.子類(lèi)是對(duì)其父類(lèi)的職責(zé)(responsibility)進(jìn)行擴(kuò)展,而非重寫(xiě)或廢除(nullify)。

            d.子類(lèi)沒(méi)有對(duì)那些僅作為一個(gè)工具類(lèi)(utility class)的功能進(jìn)行擴(kuò)展。

            e.對(duì)于一個(gè)位于實(shí)際的問(wèn)題域(Problem Domain)的類(lèi)而言,其子類(lèi)特指一種角色(role),交易(transaction)或設(shè)備(device)。





          • 繼承/組合示例1







          1."是一個(gè)…的特殊類(lèi)型",而非"是一個(gè)由…所扮演的角色"

          -->失敗。乘客是人所扮演的一種角色。代理人亦然。

          2.永遠(yuǎn)不需要轉(zhuǎn)化

          -->失敗。隨著時(shí)間的發(fā)展,一個(gè)Person的子類(lèi)實(shí)例可能會(huì)從Passenger轉(zhuǎn)變成Agent,再到Agent Passenger。

          3.擴(kuò)展,而非重寫(xiě)和廢除

          -->通過(guò)。

          4.不要擴(kuò)展一個(gè)工具類(lèi)

          -->通過(guò)。

          5.在問(wèn)題域內(nèi),特指一種角色,交易或設(shè)備

          -->失敗。Person不是一種角色,交易或設(shè)備。






          繼承并非適用于此處!


          使用組合進(jìn)行挽救!





          • 繼承/組合示例2







          1."是一個(gè)…的特殊類(lèi)型",而非"是一個(gè)由…所扮演的角色"

          -->通過(guò)。乘客和代理人都是特殊類(lèi)型的人所扮演的角色。

          2.永遠(yuǎn)不需要轉(zhuǎn)化

          -->通過(guò)。一個(gè)Passenger對(duì)象將保持不變;Agent對(duì)象亦然。

          3.擴(kuò)展,而非重寫(xiě)和廢除

          -->通過(guò)。

          4.不要擴(kuò)展一個(gè)工具類(lèi)

          -->通過(guò)。

          5.在問(wèn)題域內(nèi),特指一種角色,交易或設(shè)備

          -->通過(guò)。PersonRole是一種類(lèi)型的角色。




          繼承適用于此處!




          • 繼承/組合示例3







          1."是一個(gè)…的特殊類(lèi)型",而非"是一個(gè)由…所扮演的角色"

          -->通過(guò)。預(yù)訂和購(gòu)買(mǎi)都是一種特殊類(lèi)型的交易。

          2.永遠(yuǎn)不需要轉(zhuǎn)化

          -->通過(guò)。一個(gè)Reservation對(duì)象將保持不變;Purchase對(duì)象亦然。

          3.擴(kuò)展,而非重寫(xiě)和廢除

          -->通過(guò)。

          4.不要擴(kuò)展一個(gè)工具類(lèi)

          -->通過(guò)。

          5.在問(wèn)題域內(nèi),特指一種角色,交易或設(shè)備

          -->通過(guò)。是一種交易。




          繼承適用于此處!




          • 繼承/組合示例4







          1."是一個(gè)…的特殊類(lèi)型",而非"是一個(gè)由…所扮演的角色"

          -->失敗。預(yù)訂不是一種特殊類(lèi)型的observable。

          2.永遠(yuǎn)不需要轉(zhuǎn)化

          -->通過(guò)。一個(gè)Reservation對(duì)象將保持不變。

          3.擴(kuò)展,而非重寫(xiě)和廢除

          -->通過(guò)。

          4.不要擴(kuò)展一個(gè)工具類(lèi)

          -->失敗。Observable就是一個(gè)工具類(lèi)。

          5.在問(wèn)題域內(nèi),特指一種角色,交易或設(shè)備

          -->不適用。Observable是一個(gè)工具類(lèi),并非一個(gè)問(wèn)題域的類(lèi)。。




          繼承并非適用于此處!




          • 繼承/組合總結(jié)





            1.組合與繼承都是重要的重用方法

            2.在OO開(kāi)發(fā)的早期,繼承被過(guò)度地使用

            3.隨著時(shí)間的發(fā)展,我們發(fā)現(xiàn)優(yōu)先使用組合可以獲得重用性與簡(jiǎn)單性更佳的設(shè)計(jì)

            4.當(dāng)然可以通過(guò)繼承,以擴(kuò)充(enlarge)可用的組合類(lèi)集(the set of composable classes)。

            5.因此組合與繼承可以一起工作

            6.但是我們的基本法則是:

            優(yōu)先使用對(duì)象組合,而非(類(lèi))繼承

            [ Favor Composition Over Inheritance ]

            法則2:針對(duì)接口編程,而非(接口的)實(shí)現(xiàn)

            [ Program To An Interface, Not An Implementation ]





            • 接口



              1.接口是一個(gè)對(duì)象在對(duì)其它的對(duì)象進(jìn)行調(diào)用時(shí)所知道的方法集合。

              2.一個(gè)對(duì)象可以有多個(gè)接口(實(shí)際上,接口是對(duì)象所有方法的一個(gè)子集)

              3.類(lèi)型是對(duì)象的一個(gè)特定的接口。

              4.不同的對(duì)象可以具有相同的類(lèi)型,而且一個(gè)對(duì)象可以具有多個(gè)不同的類(lèi)型。

              5.一個(gè)對(duì)象僅能通過(guò)其接口才會(huì)被其它對(duì)象所了解。

              6.某種意義上,接口是以一種非常局限的方式,將"是一種…"表達(dá)為"一種支持該接口的…"。

              7.接口是實(shí)現(xiàn)插件化(pluggability)的關(guān)鍵





            • 實(shí)現(xiàn)繼承和接口繼承



              1.實(shí)現(xiàn)繼承(類(lèi)繼承):一個(gè)對(duì)象的實(shí)現(xiàn)是根據(jù)另一個(gè)對(duì)象的實(shí)現(xiàn)來(lái)定義的。

              2.接口繼承(子類(lèi)型化):描述了一個(gè)對(duì)象可在什么時(shí)候被用來(lái)替代另一個(gè)對(duì)象。

              3.C++的繼承機(jī)制既指類(lèi)繼承,又指接口繼承。

              4.C++通過(guò)繼承純虛類(lèi)來(lái)實(shí)現(xiàn)接口繼承。

              5.Java對(duì)接口繼承具有單獨(dú)的語(yǔ)言構(gòu)造方式-Java接口。

              6.Java接口構(gòu)造方式更加易于表達(dá)和實(shí)現(xiàn)那些專(zhuān)注于對(duì)象接口的設(shè)計(jì)。





            • 接口的好處



              1.優(yōu)點(diǎn)

              a.Client不必知道其使用對(duì)象的具體所屬類(lèi)。

              b.一個(gè)對(duì)象可以很容易地被(實(shí)現(xiàn)了相同接口的)的另一個(gè)對(duì)象所替換。

              c.對(duì)象間的連接不必硬綁定(hardwire)到一個(gè)具體類(lèi)的對(duì)象上,因此增加了靈活性。

              e.松散藕合(loosens coupling)。

              f.增加了重用的可能性。

              e.提高了(對(duì)象)組合的機(jī)率,因?yàn)楸话瑢?duì)象可以是任何實(shí)現(xiàn)了一個(gè)指定接口的類(lèi)。

              2.缺點(diǎn):

              a.設(shè)計(jì)的復(fù)雜性略有增加

              (譯者注:接口表示"…像…"(LikeA)的關(guān)系,繼承表示"…是…"(IsA)的關(guān)系,組合表示"…有…"(HasA)的關(guān)系。)





            • 接口實(shí)例








            該方法是指其它的一些類(lèi)可以進(jìn)行交通工具的駕駛,而不必關(guān)心其實(shí)際上是(汽車(chē),輪船,潛艇或是其它任何實(shí)現(xiàn)了IManeuverabre的對(duì)象)。


            法則3:開(kāi)放-封閉法則(OCP)

            軟件組成實(shí)體應(yīng)該是可擴(kuò)展的,但是不可修改的。

            [ Software Entities Should Be Open For Extension, Yet Closed For Modification
            ]




            • 開(kāi)放-封閉法則





              1.開(kāi)放-封閉法則認(rèn)為我們應(yīng)該試圖去設(shè)計(jì)出永遠(yuǎn)也不需要改變的模塊。

              2我們可以添加新代碼來(lái)擴(kuò)展系統(tǒng)的行為。我們不能對(duì)已有的代碼進(jìn)行修改。

              3.符合OCP的模塊需滿足兩個(gè)標(biāo)準(zhǔn):

              4.可擴(kuò)展,即"對(duì)擴(kuò)展是開(kāi)放的"(Open For Extension)-模塊的行為可以被擴(kuò)展,以需要滿足新的需求。

              5.不可更改,即"對(duì)更改是封閉的"(Closed for Modification)-模塊的源代碼是不允許進(jìn)行改動(dòng)的。

              6.我們能如何去做呢?

              a.抽象(Abstraction)

              b.多態(tài)(Polymorphism)

              c.繼承(Inheritance)

              d.接口(Interface)


              7. 一個(gè)軟件系統(tǒng)的所有模塊不可能都滿足OCP,但是我們應(yīng)該努力最小化這些不滿足OCP的模塊數(shù)量。

              8.開(kāi)放-封閉法則是OO設(shè)計(jì)的真正核心。

              9.符合該法則便意味著最高等級(jí)的復(fù)用性(reusability)和可維護(hù)性(maintainability)。





            • OCP示例



              1. 考慮下面某類(lèi)的方法:





              2.以上函數(shù)的工作是在制訂的部件數(shù)組中計(jì)算各個(gè)部件價(jià)格的總和。

              3.若Part是一個(gè)基類(lèi)或接口且使用了多態(tài),則該類(lèi)可很容易地來(lái)適應(yīng)新類(lèi)型的部件,而不必對(duì)其進(jìn)行修改。

              4.其將符合OCP


              5. 但是在計(jì)算總價(jià)格時(shí),若財(cái)務(wù)部頒布主板和內(nèi)存應(yīng)使用額外費(fèi)用,則將如何去做。

              6.下列的代碼是如何來(lái)做的呢?





              7.這符合OCP嗎?

              8.當(dāng)每次財(cái)務(wù)部提出新的計(jì)價(jià)策略,我們都不得不要修改totalPrice()方法!這并非"對(duì)更改是封閉的"。顯然,策略的變更便意味著我們不得不要在一些地方修改代碼的,因此我們?cè)撊绾稳プ瞿兀?br>
              9.為了使用我們第一個(gè)版本的totalPrice(),我們可以將計(jì)價(jià)策略合并到Part的getPrice()方法中。


              10.這里是Part和ConcretePart類(lèi)的示例:



              11. 但是現(xiàn)在每當(dāng)計(jì)價(jià)策略發(fā)生改變,我們就必須修改Part的每個(gè)子類(lèi)!

              12.一個(gè)更好的思路是采用一個(gè)PricePolicy類(lèi),通過(guò)對(duì)其進(jìn)行繼承以提供不同的計(jì)價(jià)策略:




              13.看起來(lái)我們所做的就是將問(wèn)題推遲到另一個(gè)類(lèi)中。但是使用該解決方案,我們可通過(guò)改變Part對(duì)象,在運(yùn)行期間動(dòng)態(tài)地來(lái)設(shè)定計(jì)價(jià)的策略。

              14.另一個(gè)解決方案是使每個(gè)ConcretePart從數(shù)據(jù)庫(kù)或?qū)傩晕募蝎@取其當(dāng)前的價(jià)格。





            • 單選法則



              單選法則(the Single Choice Principle)是OCP的一個(gè)推論。

              無(wú)論在什么時(shí)候,一個(gè)軟件系統(tǒng)必須支持一組備選項(xiàng),理想情況下,在系統(tǒng)中只能有一個(gè)類(lèi)能夠知道整個(gè)的備選項(xiàng)集合。


              法則4:Liskov替換法則(LSP)

              使用指向基類(lèi)(超類(lèi))的引用的函數(shù),必須能夠在不知道具體派生類(lèi)(子類(lèi))對(duì)象類(lèi)型的情況下使用它們。

              [ Function Thar Use Referennces To Base(Super) Classes Must Be Able To Use Objects
              Of Derived(Sub) Classes Without Knowing It ]




              • Liskov替換法則





                1.顯而易見(jiàn),Liskov替換法則(LSP)是根據(jù)我所熟知的"多態(tài)"而得出的。

                2.例如:



                方法drawShape應(yīng)該可與Sharp超類(lèi)的任何子類(lèi)一起工作(或者,若Sharp為Java接口,則該方法可與任何實(shí)現(xiàn)了Sharp接口的類(lèi)一起工作)

                但是當(dāng)我們?cè)趯?shí)現(xiàn)子類(lèi)時(shí)必須要謹(jǐn)慎對(duì)待,以確保我們不會(huì)無(wú)意中違背了LSP。





                3.若一個(gè)函數(shù)未能滿足LSP,那么可能是因?yàn)樗@式地引用了超類(lèi)的一些或所有子類(lèi)。這樣的函數(shù)也違背了OCP,因?yàn)楫?dāng)我們創(chuàng)建一個(gè)新的子類(lèi)時(shí),會(huì)不得不進(jìn)行代碼的修改。




              • LSP示例



                1. 考慮下面Rectangle類(lèi):



                2.現(xiàn)在,Square類(lèi)會(huì)如何呢?顯然,一個(gè)正方形是一個(gè)四邊形,因此Square類(lèi)應(yīng)該從Rectangle類(lèi)派生而來(lái),對(duì)否?讓我們看一看!

                3.觀察可得:

                a.正方形不需要將高和寬都作為屬性,但是總之它將繼承自Rectangle。因此,每一個(gè)Square對(duì)象會(huì)浪費(fèi)一點(diǎn)內(nèi)存,但這并不是一個(gè)主要問(wèn)題。

                b.繼承而來(lái)的setWidth()和setHeight()方法對(duì)于Square而言并非真正地適合,因?yàn)橐粋€(gè)正方形的高和寬是相同。因此我們將需要重寫(xiě)setWidth()和setHeight()方法。不得不重寫(xiě)這些簡(jiǎn)單的方法有可能是一種不恰當(dāng)?shù)睦^承使用方式。


                3.Square類(lèi)如下:










              4. 看起來(lái)都還不錯(cuò)。但是讓我們檢驗(yàn)一下!








              5. 測(cè)試程序輸出:







              6.看上去好像我們違背了LSP!


              7.這里的問(wèn)題出在哪里呢?編寫(xiě)testLsp()方法的程序員做了一個(gè)合理的假設(shè),即改變Rectangle的寬而保持它的高不變。

              8.在將一個(gè)Square對(duì)象傳遞給這樣一個(gè)方法時(shí)產(chǎn)生了問(wèn)題,顯然是違背了LSP

              9.Square和Rectangle類(lèi)是相互一致和合法的。盡管程序員對(duì)基類(lèi)作了合理的假設(shè),但其所編寫(xiě)的方法仍然會(huì)導(dǎo)致設(shè)計(jì)模型的失敗。

              10.不能孤立地去看待解決方案,必須根據(jù)設(shè)計(jì)用戶所做的合理假設(shè)來(lái)看待它們。


              11. 一個(gè)數(shù)學(xué)意義上的正方形可能是一個(gè)四邊形,但是一個(gè)Square對(duì)象不是一個(gè)Rectangle對(duì)象,因?yàn)橐粋€(gè)Square對(duì)象的行為與一個(gè)Rectangle對(duì)象的行為是不一致的!

              12.從行為上來(lái)說(shuō),一個(gè)Square不是一個(gè)Rectangle!一個(gè)Square對(duì)象與一個(gè)Rectangle對(duì)象之間不具有多態(tài)的特征。






              • 總結(jié)





                1.Liskov替換法則(LSP)清楚地表明了ISA關(guān)系全部都是與行為有關(guān)的。

                2.為了保持LSP(并與開(kāi)放-封閉法則一起),所有子類(lèi)必須符合使用基類(lèi)的client所期望的行為。

                3.一個(gè)子類(lèi)型不得具有比基類(lèi)型(base type)更多的限制,可能這對(duì)于基類(lèi)型來(lái)說(shuō)是合法的,但是可能會(huì)因?yàn)檫`背子類(lèi)型的其中一個(gè)額外限制,從而違背了LSP!

                4.LSP保證一個(gè)子類(lèi)總是能夠被用在其基類(lèi)可以出現(xiàn)的地方!

          posted @ 2007-07-02 13:11 和田雨 閱讀(215) | 評(píng)論 (0)編輯 收藏

               摘要: 摘要:?jiǎn)吸c(diǎn)登錄(SSO)的技術(shù)被越來(lái)越廣泛地運(yùn)用到各個(gè)領(lǐng)域的軟件系統(tǒng)當(dāng)中。本文從業(yè)務(wù)的角度分析了單點(diǎn)登錄的需求和應(yīng)用領(lǐng)域;從技術(shù)本身的角度分析了單點(diǎn)登錄技術(shù)的內(nèi)部機(jī)制和實(shí)現(xiàn)手段,并且給出Web-SSO和桌面SSO的實(shí)現(xiàn)、源代碼和詳細(xì)講解;還從安全和性能的角度對(duì)現(xiàn)有的實(shí)現(xiàn)技術(shù)進(jìn)行進(jìn)一步分析,指出相應(yīng)的風(fēng)險(xiǎn)和需要改進(jìn)的方面。本文除了從多個(gè)方面和角度給出了對(duì)單點(diǎn)登錄(SSO)的全面分析,還并且討論了如何...  閱讀全文
          posted @ 2007-07-02 13:10 和田雨 閱讀(640) | 評(píng)論 (0)編輯 收藏

               摘要: JDK 1.4以前,Java的IO操作集中在java.io這個(gè)包中,是基于流的阻塞(blocking)API。對(duì)于大多數(shù)應(yīng)用來(lái)說(shuō),這樣的API使用很方便,然而,一些對(duì)性能要求較高的應(yīng)用,尤其是服務(wù)端應(yīng)用,往往需要一個(gè)更為有效的方式來(lái)處理IO。從JDK 1.4起,NIO API作為一個(gè)基于緩沖區(qū),并能提供非阻塞(non-blocking)IO操作的API被引入。本文對(duì)其進(jìn)行深入的介紹。  ...  閱讀全文
          posted @ 2007-07-02 13:07 和田雨 閱讀(263) | 評(píng)論 (0)編輯 收藏

          1 定義頭和根元素

          部署描述符文件就像所有XML文件一樣,必須以一個(gè)XML頭開(kāi)始。這個(gè)頭聲明可以使用的XML版本并給出文件的字符編碼。
          DOCYTPE聲明必須立即出現(xiàn)在此頭之后。這個(gè)聲明告訴服務(wù)器適用的servlet規(guī)范的版本(如2.2或2.3)并指定管理此文件其余部分內(nèi)容的語(yǔ)法的DTD(Document Type Definition,文檔類(lèi)型定義)。
          所有部署描述符文件的頂層(根)元素為web-app。請(qǐng)注意,XML元素不像HTML,他們是大小寫(xiě)敏感的。因此,web-App和WEB-APP都是不合法的,web-app必須用小寫(xiě)。

          1 定義頭和根元素

          XML 元素不僅是大小寫(xiě)敏感的,而且它們還對(duì)出現(xiàn)在其他元素中的次序敏感。例如,XML頭必須是文件中的第一項(xiàng),DOCTYPE聲明必須是第二項(xiàng),而web- app元素必須是第三項(xiàng)。在web-app元素內(nèi),元素的次序也很重要。服務(wù)器不一定強(qiáng)制要求這種次序,但它們?cè)试S(實(shí)際上有些服務(wù)器就是這樣做的)完全 拒絕執(zhí)行含有次序不正確的元素的Web應(yīng)用。這表示使用非標(biāo)準(zhǔn)元素次序的web.xml文件是不可移植的。
          下面的列表給出了所有可直接出現(xiàn)在web-app元素內(nèi)的合法元素所必需的次序。例如,此列表說(shuō)明servlet元素必須出現(xiàn)在所有servlet-mapping元素之前。請(qǐng)注意,所有這些元素都是可選的。因此,可以省略掉某一元素,但不能把它放于不正確的位置。
          l icon icon元素指出IDE和GUI工具用來(lái)表示W(wǎng)eb應(yīng)用的一個(gè)和兩個(gè)圖像文件的位置。
          l display-name display-name元素提供GUI工具可能會(huì)用來(lái)標(biāo)記這個(gè)特定的Web應(yīng)用的一個(gè)名稱(chēng)。
          l description description元素給出與此有關(guān)的說(shuō)明性文本。
          l context-param context-param元素聲明應(yīng)用范圍內(nèi)的初始化參數(shù)。
          l filter 過(guò)濾器元素將一個(gè)名字與一個(gè)實(shí)現(xiàn)javax.servlet.Filter接口的類(lèi)相關(guān)聯(lián)。
          l filter-mapping 一旦命名了一個(gè)過(guò)濾器,就要利用filter-mapping元素把它與一個(gè)或多個(gè)servlet或JSP頁(yè)面相關(guān)聯(lián)。
          l listener servlet API的版本2.3增加了對(duì)事件監(jiān)聽(tīng)程序的支持,事件監(jiān)聽(tīng)程序在建立、修改和刪除會(huì)話或servlet環(huán)境時(shí)得到通知。Listener元素指出事件監(jiān)聽(tīng)程序類(lèi)。
          l servlet 在向servlet或JSP頁(yè)面制定初始化參數(shù)或定制URL時(shí),必須首先命名servlet或JSP頁(yè)面。Servlet元素就是用來(lái)完成此項(xiàng)任務(wù)的。
          l servlet-mapping 服務(wù)器一般為servlet提供一個(gè)缺省的URL:http://host/webAppPrefix/servlet/ServletName。但是,常常會(huì)更改這個(gè)URL,以便servlet可以訪問(wèn)初始化參數(shù)或更容易地處理相對(duì)URL。在更改缺省URL時(shí),使用servlet-mapping元素。
          l session-config 如果某個(gè)會(huì)話在一定時(shí)間內(nèi)未被訪問(wèn),服務(wù)器可以拋棄它以節(jié)省內(nèi)存。可通過(guò)使用HttpSession的setMaxInactiveInterval方法 明確設(shè)置單個(gè)會(huì)話對(duì)象的超時(shí)值,或者可利用session-config元素制定缺省超時(shí)值。
          l mime-mapping 如果Web應(yīng)用具有想到特殊的文件,希望能保證給他們分配特定的MIME類(lèi)型,則mime-mapping元素提供這種保證。
          l welcom-file-list welcome-file-list元素指示服務(wù)器在收到引用一個(gè)目錄名而不是文件名的URL時(shí),使用哪個(gè)文件。
          l error-page error-page元素使得在返回特定HTTP狀態(tài)代碼時(shí),或者特定類(lèi)型的異常被拋出時(shí),能夠制定將要顯示的頁(yè)面。
          l taglib taglib元素對(duì)標(biāo)記庫(kù)描述符文件(Tag Libraryu Descriptor file)指定別名。此功能使你能夠更改TLD文件的位置,而不用編輯使用這些文件的JSP頁(yè)面。
          l resource-env-ref resource-env-ref元素聲明與資源相關(guān)的一個(gè)管理對(duì)象。
          l resource-ref resource-ref元素聲明一個(gè)資源工廠使用的外部資源。
          l security-constraint security-constraint元素制定應(yīng)該保護(hù)的URL。它與login-config元素聯(lián)合使用
          l login-config 用login-config元素來(lái)指定服務(wù)器應(yīng)該怎樣給試圖訪問(wèn)受保護(hù)頁(yè)面的用戶授權(quán)。它與sercurity-constraint元素聯(lián)合使用。
          l security-role security-role元素給出安全角色的一個(gè)列表,這些角色將出現(xiàn)在servlet元素內(nèi)的security-role-ref元素的role-name子元素中。分別地聲明角色可使高級(jí)IDE處理安全信息更為容易。
          l env-entry env-entry元素聲明Web應(yīng)用的環(huán)境項(xiàng)。
          l ejb-ref ejb-ref元素聲明一個(gè)EJB的主目錄的引用。
          l ejb-local-ref ejb-local-ref元素聲明一個(gè)EJB的本地主目錄的應(yīng)用。

          3 分配名稱(chēng)和定制的UL

          在web.xml中完成的一個(gè)最常見(jiàn)的任務(wù)是對(duì)servlet或JSP頁(yè)面給出名稱(chēng)和定制的URL。用servlet元素分配名稱(chēng),使用servlet-mapping元素將定制的URL與剛分配的名稱(chēng)相關(guān)聯(lián)。
          3.1 分配名稱(chēng)
          為 了提供初始化參數(shù),對(duì)servlet或JSP頁(yè)面定義一個(gè)定制URL或分配一個(gè)安全角色,必須首先給servlet或JSP頁(yè)面一個(gè)名稱(chēng)。可通過(guò) servlet元素分配一個(gè)名稱(chēng)。最常見(jiàn)的格式包括servlet-name和servlet-class子元素(在web-app元素內(nèi)),如下所示:
          <servlet>
          <servlet-name>Test</servlet-name>
          <servlet-class>moreservlets.TestServlet</servlet-class>
          </servlet> 
          這 表示位于WEB-INF/classes/moreservlets/TestServlet的servlet已經(jīng)得到了注冊(cè)名Test。給 servlet一個(gè)名稱(chēng)具有兩個(gè)主要的含義。首先,初始化參數(shù)、定制的URL模式以及其他定制通過(guò)此注冊(cè)名而不是類(lèi)名引用此servlet。其次,可在 URL而不是類(lèi)名中使用此名稱(chēng)。因此,利用剛才給出的定義,URL http://host/webAppPrefix/servlet/Test 可用于 http://host/webAppPrefix/servlet/moreservlets.TestServlet 的場(chǎng)所。
          請(qǐng) 記住:XML元素不僅是大小寫(xiě)敏感的,而且定義它們的次序也很重要。例如,web-app元素內(nèi)所有servlet元素必須位于所有servlet- mapping元素(下一小節(jié)介紹)之前,而且還要位于5.6節(jié)和5.11節(jié)討論的與過(guò)濾器或文檔相關(guān)的元素(如果有的話)之前。類(lèi)似地,servlet 的servlet-name子元素也必須出現(xiàn)在servlet-class之前。5.2節(jié)"部署描述符文件內(nèi)的元素次序"將詳細(xì)介紹這種必需的次序。
          例 如,程序清單5-1給出了一個(gè)名為T(mén)estServlet的簡(jiǎn)單servlet,它駐留在moreservlets程序包中。因?yàn)榇藄ervlet是扎根 在一個(gè)名為deployDemo的目錄中的Web應(yīng)用的組成部分,所以TestServlet.class放在deployDemo/WEB- INF/classes/moreservlets中。程序清單5-2給出將放置在deployDemo/WEB-INF/內(nèi)的web.xml文件的一部 分。此web.xml文件使用servlet-name和servlet-class元素將名稱(chēng)Test與TestServlet.class相關(guān)聯(lián)。圖 5-1和圖5-2分別顯示利用缺省URL和注冊(cè)名調(diào)用TestServlet時(shí)的結(jié)果。

          程序清單5-1 TestServlet.java
          package moreservlets;

          import java.io.*;
          import javax.servlet.*;
          import javax.servlet.http.*;

          /** Simple servlet used to illustrate servlet naming
          * and custom URLs.
          * <P>
          * Taken from More Servlets and JavaServer Pages
          * from Prentice Hall and Sun Microsystems Press,
          http://www.moreservlets.com/.
          * © 2002 Marty Hall; may be freely used or adapted.
          */

          public class TestServlet extends HttpServlet {
          public void doGet(HttpServletRequest request,
          HttpServletResponse response)
          throws ServletException, IOException {
          response.setContentType("text/html");
          PrintWriter out = response.getWriter();
          String uri = request.getRequestURI();
          out.println(ServletUtilities.headWithTitle("Test Servlet") +
          "<BODY BGCOLOR=\"#FDF5E6\">\n" +
          "<H2>URI: " + uri + "</H2>\n" +
          "</BODY></HTML>");
          }
          }


          程序清單5-2 web.xml(說(shuō)明servlet名稱(chēng)的摘錄)
          <?xml version="1.0" encoding="ISO-8859-1"?>
          <!DOCTYPE web-app
          PUBLIC "-//Sun Microsystems, Inc.//DTD Web Application 2.3//EN"
          "http://java.sun.com/dtd/web-app_2_3.dtd">

          <web-app>
          <!-- … -->
          <servlet>
          <servlet-name>Test</servlet-name>
          <servlet-class>moreservlets.TestServlet</servlet-class>
          </servlet>
          <!-- … -->
          </web-app>

          3.2 定義定制的URL
          大多數(shù)服務(wù)器具有一個(gè)缺省的serlvet URL:
          http://host/webAppPrefix/servlet/packageName.ServletName。 雖然在開(kāi)發(fā)中使用這個(gè)URL很方便,但是我們常常會(huì)希望另一個(gè)URL用于部署。例如,可能會(huì)需要一個(gè)出現(xiàn)在Web應(yīng)用頂層的URL(如,http: //host/webAppPrefix/Anyname),并且在此URL中沒(méi)有servlet項(xiàng)。位于頂層的URL簡(jiǎn)化了相對(duì)URL的使用。此外,對(duì) 許多開(kāi)發(fā)人員來(lái)說(shuō),頂層URL看上去比更長(zhǎng)更麻煩的缺省URL更簡(jiǎn)短。
          事實(shí)上,有時(shí)需要使用定制的URL。比如,你可能想關(guān)閉缺省URL映射,以便更好地強(qiáng)制實(shí)施安全限制或防止用戶意外地訪問(wèn)無(wú)初始化參數(shù)的servlet。如果你禁止了缺省的URL,那么你怎樣訪問(wèn)servlet呢?這時(shí)只有使用定制的URL了。
          為 了分配一個(gè)定制的URL,可使用servlet-mapping元素及其servlet-name和url-pattern子元素。Servlet- name元素提供了一個(gè)任意名稱(chēng),可利用此名稱(chēng)引用相應(yīng)的servlet;url-pattern描述了相對(duì)于Web應(yīng)用的根目錄的URL。url- pattern元素的值必須以斜杠(/)起始。
          下面給出一個(gè)簡(jiǎn)單的web.xml摘錄,它允許使用URL http://host/webAppPrefix/UrlTest而不是http://host/webAppPrefix/servlet/Test或
          http: //host/webAppPrefix/servlet/moreservlets.TestServlet。請(qǐng)注意,仍然需要XML頭、 DOCTYPE聲明以及web-app封閉元素。此外,可回憶一下,XML元素出現(xiàn)地次序不是隨意的。特別是,需要把所有servlet元素放在所有 servlet-mapping元素之前。
          <servlet>
          <servlet-name>Test</servlet-name>
          <servlet-class>moreservlets.TestServlet</servlet-class>
          </servlet>
          <!-- ... -->
          <servlet-mapping>
          <servlet-name>Test</servlet-name>
          <url-pattern>/UrlTest</url-pattern>
          </servlet-mapping>
          URL模式還可以包含通配符。例如,下面的小程序指示服務(wù)器發(fā)送所有以Web應(yīng)用的URL前綴開(kāi)始,以..asp結(jié)束的請(qǐng)求到名為BashMS的servlet。
          <servlet>
          <servlet-name>BashMS</servlet-name>
          <servlet-class>msUtils.ASPTranslator</servlet-class>
          </servlet>
          <!-- ... -->
          <servlet-mapping>
          <servlet-name>BashMS</servlet-name>
          <url-pattern>/*.asp</url-pattern>
          </servlet-mapping>
          3.3 命名JSP頁(yè)面
          因 為JSP頁(yè)面要轉(zhuǎn)換成sevlet,自然希望就像命名servlet一樣命名JSP頁(yè)面。畢竟,JSP頁(yè)面可能會(huì)從初始化參數(shù)、安全設(shè)置或定制的URL中 受益,正如普通的serlvet那樣。雖然JSP頁(yè)面的后臺(tái)實(shí)際上是servlet這句話是正確的,但存在一個(gè)關(guān)鍵的猜疑:即,你不知道JSP頁(yè)面的實(shí)際 類(lèi)名(因?yàn)橄到y(tǒng)自己挑選這個(gè)名字)。因此,為了命名JSP頁(yè)面,可將jsp-file元素替換為servlet-calss元素,如下所示:
          <servlet>
          <servlet-name>Test</servlet-name>
          <jsp-file>/TestPage.jsp</jsp-file>
          </servlet>
          命 名JSP頁(yè)面的原因與命名servlet的原因完全相同:即為了提供一個(gè)與定制設(shè)置(如,初始化參數(shù)和安全設(shè)置)一起使用的名稱(chēng),并且,以便能更改激活 JSP頁(yè)面的URL(比方說(shuō),以便多個(gè)URL通過(guò)相同頁(yè)面得以處理,或者從URL中去掉.jsp擴(kuò)展名)。但是,在設(shè)置初始化參數(shù)時(shí),應(yīng)該注意,JSP頁(yè) 面是利用jspInit方法,而不是init方法讀取初始化參數(shù)的。
          例如,程序清單5-3給出一個(gè)名為T(mén)estPage.jsp的簡(jiǎn)單JSP頁(yè)面,它的工作只是打印出用來(lái)激活它的URL的本地部分。TestPage.jsp放置在deployDemo應(yīng)用的頂層。程序清單5-4給出了用來(lái)分配一個(gè)注冊(cè)名PageName,然后將此注冊(cè)名與http://host/webAppPrefix/UrlTest2/anything 形式的URL相關(guān)聯(lián)的web.xml文件(即,deployDemo/WEB-INF/web.xml)的一部分。

          程序清單5-3 TestPage.jsp
          <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0 Transitional//EN">
          <HTML>
          <HEAD>
          <TITLE>
          JSP Test Page
          </TITLE>
          </HEAD>
          <BODY BGCOLOR="#FDF5E6">
          <H2>URI: <%= request.getRequestURI() %></H2>
          </BODY>
          </HTML>


          程序清單5-4 web.xml(說(shuō)明JSP頁(yè)命名的摘錄)
          <?xml version="1.0" encoding="ISO-8859-1"?>
          <!DOCTYPE web-app
          PUBLIC "-//Sun Microsystems, Inc.//DTD Web Application 2.3//EN"
          "http://java.sun.com/dtd/web-app_2_3.dtd">

          <web-app>
          <!-- ... -->
          <servlet>
          <servlet-name>PageName</servlet-name>
          <jsp-file>/TestPage.jsp</jsp-file>
          </servlet>
          <!-- ... -->
          <servlet-mapping>
          <servlet-name> PageName </servlet-name>
          <url-pattern>/UrlTest2/*</url-pattern>
          </servlet-mapping>
          <!-- ... -->
          </web-app>


          4 禁止激活器servlet

          對(duì)servlet 或JSP頁(yè)面建立定制URL的一個(gè)原因是,這樣做可以注冊(cè)從 init(servlet)或jspInit(JSP頁(yè)面)方法中讀取得初始化參數(shù)。但是,初始化參數(shù)只在是利用定制URL模式或注冊(cè)名訪問(wèn) servlet或JSP頁(yè)面時(shí)可以使用,用缺省URL http://host/webAppPrefix/servlet/ServletName 訪問(wèn)時(shí)不能使用。因此,你可能會(huì)希望關(guān)閉缺省URL,這樣就不會(huì)有人意外地調(diào)用初始化servlet了。這個(gè)過(guò)程有時(shí)稱(chēng)為禁止激活器servlet,因?yàn)?nbsp;多數(shù)服務(wù)器具有一個(gè)用缺省的servlet URL注冊(cè)的標(biāo)準(zhǔn)servlet,并激活缺省的URL應(yīng)用的實(shí)際servlet。
          有兩種禁止此缺省URL的主要方法:
          l 在每個(gè)Web應(yīng)用中重新映射/servlet/模式。
          l 全局關(guān)閉激活器servlet。
          重 要的是應(yīng)該注意到,雖然重新映射每個(gè)Web應(yīng)用中的/servlet/模式比徹底禁止激活servlet所做的工作更多,但重新映射可以用一種完全可移植 的方式來(lái)完成。相反,全局禁止激活器servlet完全是針對(duì)具體機(jī)器的,事實(shí)上有的服務(wù)器(如ServletExec)沒(méi)有這樣的選擇。下面的討論對(duì)每 個(gè)Web應(yīng)用重新映射/servlet/ URL模式的策略。后面提供在Tomcat中全局禁止激活器servlet的詳細(xì)內(nèi)容。
          4.1 重新映射/servlet/URL模式
          在一個(gè)特定的Web應(yīng)用中禁止以http://host/webAppPrefix/servlet/ 開(kāi)始的URL的處理非常簡(jiǎn)單。所需做的事情就是建立一個(gè)錯(cuò)誤消息servlet,并使用前一節(jié)討論的url-pattern元素將所有匹配請(qǐng)求轉(zhuǎn)向該 servlet。只要簡(jiǎn)單地使用:
          <url-pattern>/servlet/*</url-pattern>
          作為servlet-mapping元素中的模式即可。
          例如,程序清單5-5給出了將SorryServlet servlet(程序清單5-6)與所有以http://host/webAppPrefix/servlet/ 開(kāi)頭的URL相關(guān)聯(lián)的部署描述符文件的一部分。

          程序清單5-5 web.xml(說(shuō)明JSP頁(yè)命名的摘錄)
          <?xml version="1.0" encoding="ISO-8859-1"?>
          <!DOCTYPE web-app
          PUBLIC "-//Sun Microsystems, Inc.//DTD Web Application 2.3//EN"
          "http://java.sun.com/dtd/web-app_2_3.dtd">

          <web-app>
          <!-- ... -->
          <servlet>
          <servlet-name>Sorry</servlet-name>
          <servlet-class>moreservlets.SorryServlet</servlet-class>
          </servlet>
          <!-- ... -->
          <servlet-mapping>
          <servlet-name> Sorry </servlet-name>
          <url-pattern>/servlet/*</url-pattern>
          </servlet-mapping>
          <!-- ... -->
          </web-app>


          程序清單5-6 SorryServlet.java
          package moreservlets;

          import java.io.*;
          import javax.servlet.*;
          import javax.servlet.http.*;

          /** Simple servlet used to give error messages to
          * users who try to access default servlet URLs
          * (i.e., http://host/webAppPrefix/servlet/ServletName)
          * in Web applications that have disabled this
          * behavior.
          * <P>
          * Taken from More Servlets and JavaServer Pages
          * from Prentice Hall and Sun Microsystems Press,
          * http://www.moreservlets.com/.
          * © 2002 Marty Hall; may be freely used or adapted.
          */

          public class SorryServlet extends HttpServlet {
          public void doGet(HttpServletRequest request,
          HttpServletResponse response)
          throws ServletException, IOException {
          response.setContentType("text/html");
          PrintWriter out = response.getWriter();
          String title = "Invoker Servlet Disabled.";
          out.println(ServletUtilities.headWithTitle(title) +
          "<BODY BGCOLOR=\"#FDF5E6\">\n" +
          "<H2>" + title + "</H2>\n" +
          "Sorry, access to servlets by means of\n" +
          "URLs that begin with\n" +
          "http://host/webAppPrefix/servlet/\n" +
          "has been disabled.\n" + 
          "</BODY></HTML>");
          }

          public void doPost(HttpServletRequest request,
          HttpServletResponse response)
          throws ServletException, IOException {
          doGet(request, response);
          }
          }


          4.2 全局禁止激活器:Tomcat
          Tomcat 4中用來(lái)關(guān)閉缺省URL的方法與Tomcat 3中所用的很不相同。下面介紹這兩種方法:
          1.禁止激活器: Tomcat 4
          Tomcat 4用與前面相同的方法關(guān)閉激活器servlet,即利用web.xml中的url-mapping元素進(jìn)行關(guān)閉。不同之處在于Tomcat使用了放在 install_dir/conf中的一個(gè)服務(wù)器專(zhuān)用的全局web.xml文件,而前面使用的是存放在每個(gè)Web應(yīng)用的WEB-INF目錄中的標(biāo)準(zhǔn) web.xml文件。
          因此,為了在Tomcat 4中關(guān)閉激活器servlet,只需在install_dir/conf/web.xml中簡(jiǎn)單地注釋出/servlet/* URL映射項(xiàng)即可,如下所示:
          <!-- 
          <servlet-mapping>
          <servlet-name>invoker</servlet-name>
          <url-pattern>/servlet/*</url-pattern>
          </servlet-mapping>
          -->
          再次提醒,應(yīng)該注意這個(gè)項(xiàng)是位于存放在install_dir/conf的Tomcat專(zhuān)用的web.xml文件中的,此文件不是存放在每個(gè)Web應(yīng)用的WEB-INF目錄中的標(biāo)準(zhǔn)web.xml。
          2.禁止激活器:Tomcat3
          在Apache Tomcat的版本3中,通過(guò)在install_dir/conf/server.xml中注釋出InvokerInterceptor項(xiàng)全局禁止缺省 servlet URL。例如,下面是禁止使用缺省servlet URL的server.xml文件的一部分。
          <!-- 
          <RequsetInterceptor 
          className="org.apache.tomcat.request.InvokerInterceptor"
          debug="0" prefix="/servlet/" />
          -->

          5 初始化和預(yù)裝載servlet與JSP頁(yè)面

          這里討論控制servlet和JSP頁(yè)面的啟動(dòng)行為的方法。特別是,說(shuō)明了怎樣分配初始化參數(shù)以及怎樣更改服務(wù)器生存期中裝載servlet和JSP頁(yè)面的時(shí)刻。
          5.1 分配servlet初始化參數(shù)
          利 用init-param元素向servlet提供初始化參數(shù),init-param元素具有param-name和param-value子元素。例如, 在下面的例子中,如果initServlet servlet是利用它的注冊(cè)名(InitTest)訪問(wèn)的,它將能夠從其方法中調(diào)用getServletConfig(). getInitParameter("param1")獲得"Value 1",調(diào)用getServletConfig().getInitParameter("param2")獲得"2"。
          <servlet>
          <servlet-name>InitTest</servlet-name>
          <servlet-class>moreservlets.InitServlet</servlet-class>
          <init-param>
          <param-name>param1</param-name>
          <param-value>value1</param-value>
          </init-param>
          <init-param>
          <param-name>param2</param-name>
          <param-value>2</param-value>
          </init-param>
          </servlet>
          在涉及初始化參數(shù)時(shí),有幾點(diǎn)需要注意:
          l 返回值。GetInitParameter的返回值總是一個(gè)String。因此,在前一個(gè)例子中,可對(duì)param2使用Integer.parseInt獲得一個(gè)int。
          l JSP中的初始化。JSP頁(yè)面使用jspInit而不是init。JSP頁(yè)面還需要使用jsp-file元素代替servlet-class。
          l 缺省URL。初始化參數(shù)只在通過(guò)它們的注冊(cè)名或與它們注冊(cè)名相關(guān)的定制URL模式訪問(wèn)Servlet時(shí)可以使用。因此,在這個(gè)例子中,param1和 param2初始化參數(shù)將能夠在使用URL http://host/webAppPrefix/servlet/InitTest 時(shí)可用,但在使用URL http://host/webAppPrefix/servlet/myPackage.InitServlet 時(shí)不能使用。
          例如,程序清單5-7給出一個(gè)名為InitServlet的簡(jiǎn)單servlet,它使用init方法設(shè)置firstName和emailAddress字段。程序清單5-8給出分配名稱(chēng)InitTest給servlet的web.xml文件。
          程序清單5-7 InitServlet.java
          package moreservlets;

          import java.io.*;
          import javax.servlet.*;
          import javax.servlet.http.*;

          /** Simple servlet used to illustrate servlet
          * initialization parameters.
          * <P>
          * Taken from More Servlets and JavaServer Pages
          * from Prentice Hall and Sun Microsystems Press,
          * http://www.moreservlets.com/.
          * © 2002 Marty Hall; may be freely used or adapted.
          */

          public class InitServlet extends HttpServlet {
          private String firstName, emailAddress;

          public void init() {
          ServletConfig config = getServletConfig();
          firstName = config.getInitParameter("firstName");
          emailAddress = config.getInitParameter("emailAddress");
          }

          public void doGet(HttpServletRequest request,
          HttpServletResponse response)
          throws ServletException, IOException {
          response.setContentType("text/html");
          PrintWriter out = response.getWriter();
          String uri = request.getRequestURI();
          out.println(ServletUtilities.headWithTitle("Init Servlet") +
          "<BODY BGCOLOR=\"#FDF5E6\">\n" +
          "<H2>Init Parameters:</H2>\n" +
          "<UL>\n" +
          "<LI>First name: " + firstName + "\n" +
          "<LI>Email address: " + emailAddress + "\n" +
          "</UL>\n" + 
          "</BODY></HTML>");
          }
          }


          程序清單5-8 web.xml(說(shuō)明初始化參數(shù)的摘錄)
          <?xml version="1.0" encoding="ISO-8859-1"?>
          <!DOCTYPE web-app
          PUBLIC "-//Sun Microsystems, Inc.//DTD Web Application 2.3//EN"
          "http://java.sun.com/dtd/web-app_2_3.dtd">

          <web-app>
          <!-- ... -->
          <servlet>
          <servlet-name>InitTest</servlet-name>
          <servlet-class>moreservlets.InitServlet</servlet-class>
          <init-param>
          <param-name>firstName</param-name>
          <param-value>Larry</param-value>
          </init-param>
          <init-param>
          <param-name>emailAddress</param-name>
          <param-value>Ellison@Microsoft.com</param-value>
          </init-param>
          </servlet>
          <!-- ... -->
          </web-app>

          5.2 分配JSP初始化參數(shù)
          給JSP頁(yè)面提供初始化參數(shù)在三個(gè)方面不同于給servlet提供初始化參數(shù)。
          1)使用jsp-file而不是servlet-class。因此,WEB-INF/web.xml文件的servlet元素如下所示:
          <servlet>
          <servlet-name>PageName</servlet-name>
          <jsp-file>/RealPage.jsp</jsp-file>
          <init-param>
          <param-name>...</param-name>
          <param-value>...</param-value>
          </init-param>
          ...
          </servlet>
          2) 幾乎總是分配一個(gè)明確的URL模式。對(duì)servlet,一般相應(yīng)地使用以http://host/webAppPrefix/servlet/ 開(kāi)始的缺省URL。只需記住,使用注冊(cè)名而不是原名稱(chēng)即可。這對(duì)于JSP頁(yè)面在技術(shù)上也是合法的。例如,在上面給出的例子中,可用URL http://host/webAppPrefix/servlet/PageName 訪問(wèn)RealPage.jsp的對(duì)初始化參數(shù)具有訪問(wèn)權(quán)的版本。但在用于JSP頁(yè)面時(shí),許多用戶似乎不喜歡應(yīng)用常規(guī)的servlet的URL。此外,如果 JSP頁(yè)面位于服務(wù)器為其提供了目錄清單的目錄中(如,一個(gè)既沒(méi)有index.html也沒(méi)有index.jsp文件的目錄),則用戶可能會(huì)連接到此 JSP頁(yè)面,單擊它,從而意外地激活未初始化的頁(yè)面。因此,好的辦法是使用url-pattern(5.3節(jié))將JSP頁(yè)面的原URL與注冊(cè)的 servlet名相關(guān)聯(lián)。這樣,客戶機(jī)可使用JSP頁(yè)面的普通名稱(chēng),但仍然激活定制的版本。例如,給定來(lái)自項(xiàng)目1的servlet定義,可使用下面的 servlet-mapping定義:
          <servlet-mapping>
          <servlet-name>PageName</servlet-name>
          <url-pattern>/RealPage.jsp</url-pattern>
          </servlet-mapping>
          3)JSP頁(yè)使用jspInit而不是init。自動(dòng)從JSP頁(yè)面建立的servlet或許已經(jīng)使用了inti方法。因此,使用JSP聲明提供一個(gè)init方法是不合法的,必須制定jspInit方法。
          為了說(shuō)明初始化JSP頁(yè)面的過(guò)程,程序清單5-9給出了一個(gè)名為InitPage.jsp的JSP頁(yè)面,它包含一個(gè)jspInit方法且放置于 deployDemo Web應(yīng)用層次結(jié)構(gòu)的頂層。一般,http://host/deployDemo/InitPage.jsp 形式的URL將激活此頁(yè)面的不具有初始化參數(shù)訪問(wèn)權(quán)的版本,從而將對(duì)firstName和emailAddress變量顯示null。但是, web.xml文件(程序清單5-10)分配了一個(gè)注冊(cè)名,然后將該注冊(cè)名與URL模式/InitPage.jsp相關(guān)聯(lián)。

          程序清單5-9 InitPage.jsp
          <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0 Transitional//EN">
          <HTML>
          <HEAD><TITLE>JSP Init Test</TITLE></HEAD>
          <BODY BGCOLOR="#FDF5E6">
          <H2>Init Parameters:</H2>
          <UL>
          <LI>First name: <%= firstName %>
          <LI>Email address: <%= emailAddress %>
          </UL>
          </BODY></HTML>
          <%!
          private String firstName, emailAddress;

          public void jspInit() {
          ServletConfig config = getServletConfig();
          firstName = config.getInitParameter("firstName");
          emailAddress = config.getInitParameter("emailAddress");
          }
          %>


          程序清單5-10 web.xml(說(shuō)明JSP頁(yè)面的init參數(shù)的摘錄)
          <?xml version="1.0" encoding="ISO-8859-1"?>
          <!DOCTYPE web-app
          PUBLIC "-//Sun Microsystems, Inc.//DTD Web Application 2.3//EN"
          "http://java.sun.com/dtd/web-app_2_3.dtd">

          <web-app>
          <!-- ... -->
          <servlet>
          <servlet-name>InitPage</servlet-name>
          <jsp-file>/InitPage.jsp</jsp-file>
          <init-param>
          <param-name>firstName</param-name>
          <param-value>Bill</param-value>
          </init-param>
          <init-param>
          <param-name>emailAddress</param-name>
          <param-value>gates@oracle.com</param-value>
          </init-param>
          </servlet>
          <!-- ... --> 
          <servlet-mapping>
          <servlet-name> InitPage</servlet-name>
          <url-pattern>/InitPage.jsp</url-pattern>
          </servlet-mapping>
          <!-- ... -->
          </web-app>


          5.3 提供應(yīng)用范圍內(nèi)的初始化參數(shù)
          一 般,對(duì)單個(gè)地servlet或JSP頁(yè)面分配初始化參數(shù)。指定的servlet或JSP頁(yè)面利用ServletConfig的 getInitParameter方法讀取這些參數(shù)。但是,在某些情形下,希望提供可由任意servlet或JSP頁(yè)面借助ServletContext 的getInitParameter方法讀取的系統(tǒng)范圍內(nèi)的初始化參數(shù)。
          可利用context-param元素聲明這些系統(tǒng)范圍內(nèi)的初始化值。context-param元素應(yīng)該包含param-name、param-value以及可選的description子元素,如下所示:
          <context-param>
          <param-name>support-email</param-name>
          <param-value>blackhole@mycompany.com</param-value>
          </context-param>
          可 回憶一下,為了保證可移植性,web.xml內(nèi)的元素必須以正確的次序聲明。但這里應(yīng)該注意,context-param元素必須出現(xiàn)任意與文檔有關(guān)的元 素(icon、display-name或description)之后及filter、filter-mapping、listener或 servlet元素之前。
          5.4 在服務(wù)器啟動(dòng)時(shí)裝載servlet
          假如servlet或JSP頁(yè)面有一個(gè)要花很長(zhǎng)時(shí)間執(zhí)行的init (servlet)或jspInit(JSP)方法。例如,假如init或jspInit方法從某個(gè)數(shù)據(jù)庫(kù)或ResourceBundle查找產(chǎn)量。這種 情況下,在第一個(gè)客戶機(jī)請(qǐng)求時(shí)裝載servlet的缺省行為將對(duì)第一個(gè)客戶機(jī)產(chǎn)生較長(zhǎng)時(shí)間的延遲。因此,可利用servlet的load-on- startup元素規(guī)定服務(wù)器在第一次啟動(dòng)時(shí)裝載servlet。下面是一個(gè)例子。
          <servlet>
          <servlet-name> … </servlet-name>
          <servlet-class> … </servlet-class> <!-- Or jsp-file -->
          <load-on-startup/>
          </servlet>
          可 以為此元素體提供一個(gè)整數(shù)而不是使用一個(gè)空的load-on-startup。想法是服務(wù)器應(yīng)該在裝載較大數(shù)目的servlet或JSP頁(yè)面之前裝載較少 數(shù)目的servlet或JSP頁(yè)面。例如,下面的servlet項(xiàng)(放置在Web應(yīng)用的WEB-INF目錄下的web.xml文件中的web-app元素 內(nèi))將指示服務(wù)器首先裝載和初始化SearchServlet,然后裝載和初始化由位于Web應(yīng)用的result目錄中的index.jsp文件產(chǎn)生的 servlet。
          <servlet>
          <servlet-name>Search</servlet-name>
          <servlet-class>myPackage.SearchServlet</servlet-class> <!-- Or jsp-file -->
          <load-on-startup>1</load-on-startup>
          </servlet>
          <servlet>
          <servlet-name>Results</servlet-name>
          <servlet-class>/results/index.jsp</servlet-class> <!-- Or jsp-file -->
          <load-on-startup>2</load-on-startup>
          </servlet>

          6 聲明過(guò)濾器

          servlet版本2.3引入了過(guò)濾器的概念。雖然所有支持servlet API版本2.3的服務(wù)器都支持過(guò)濾器,但為了使用與過(guò)濾器有關(guān)的元素,必須在web.xml中使用版本2.3的DTD。
          過(guò) 濾器可截取和修改進(jìn)入一個(gè)servlet或JSP頁(yè)面的請(qǐng)求或從一個(gè)servlet或JSP頁(yè)面發(fā)出的相應(yīng)。在執(zhí)行一個(gè)servlet或JSP頁(yè)面之前, 必須執(zhí)行第一個(gè)相關(guān)的過(guò)濾器的doFilter方法。在該過(guò)濾器對(duì)其FilterChain對(duì)象調(diào)用doFilter時(shí),執(zhí)行鏈中的下一個(gè)過(guò)濾器。如果沒(méi) 有其他過(guò)濾器,servlet或JSP頁(yè)面被執(zhí)行。過(guò)濾器具有對(duì)到來(lái)的ServletRequest對(duì)象的全部訪問(wèn)權(quán),因此,它們可以查看客戶機(jī)名、查找 到來(lái)的cookie等。為了訪問(wèn)servlet或JSP頁(yè)面的輸出,過(guò)濾器可將響應(yīng)對(duì)象包裹在一個(gè)替身對(duì)象(stand-in object)中,比方說(shuō)把輸出累加到一個(gè)緩沖區(qū)。在調(diào)用FilterChain對(duì)象的doFilter方法之后,過(guò)濾器可檢查緩沖區(qū),如有必要,就對(duì)它 進(jìn)行修改,然后傳送到客戶機(jī)。
          例如,程序清單5-11帝國(guó)難以了一個(gè)簡(jiǎn)單的過(guò)濾器,只要訪問(wèn)相關(guān)的servlet或JSP頁(yè)面,它就截取請(qǐng)求并在標(biāo)準(zhǔn)輸出上打印一個(gè)報(bào)告(開(kāi)發(fā)過(guò)程中在桌面系統(tǒng)上運(yùn)行時(shí),大多數(shù)服務(wù)器都可以使用這個(gè)過(guò)濾器)。

          程序清單5-11 ReportFilter.java
          package moreservlets;

          import java.io.*;
          import javax.servlet.*;
          import javax.servlet.http.*;
          import java.util.*;

          /** Simple filter that prints a report on the standard output 
          * whenever the associated servlet or JSP page is accessed.
          * <P>
          * Taken from More Servlets and JavaServer Pages
          * from Prentice Hall and Sun Microsystems Press,
          * http://www.moreservlets.com/.
          * © 2002 Marty Hall; may be freely used or adapted.
          */

          public class ReportFilter implements Filter {
          public void doFilter(ServletRequest request,
          ServletResponse response,
          FilterChain chain)
          throws ServletException, IOException {
          HttpServletRequest req = (HttpServletRequest)request;
          System.out.println(req.getRemoteHost() +
          " tried to access " +
          req.getRequestURL() +
          " on " + new Date() + ".");
          chain.doFilter(request,response);
          }

          public void init(FilterConfig config)
          throws ServletException {
          }

          public void destroy() {}
          }

          一 旦建立了一個(gè)過(guò)濾器,可以在web.xml中利用filter元素以及filter-name(任意名稱(chēng))、file-class(完全限定的類(lèi)名)和 (可選的)init-params子元素聲明它。請(qǐng)注意,元素在web.xml的web-app元素中出現(xiàn)的次序不是任意的;允許服務(wù)器(但不是必需的) 強(qiáng)制所需的次序,并且實(shí)際中有些服務(wù)器也是這樣做的。但這里要注意,所有filter元素必須出現(xiàn)在任意filter-mapping元素之前, filter-mapping元素又必須出現(xiàn)在所有servlet或servlet-mapping元素之前。
          例如,給定上述的ReportFilter類(lèi),可在web.xml中作出下面的filter聲明。它把名稱(chēng)Reporter與實(shí)際的類(lèi)ReportFilter(位于moreservlets程序包中)相關(guān)聯(lián)。
          <filter>
          <filter-name>Reporter</filter-name>
          <filter-class>moresevlets.ReportFilter</filter-class>
          </filter>
          一旦命名了一個(gè)過(guò)濾器,可利用filter-mapping元素把它與一個(gè)或多個(gè)servlet或JSP頁(yè)面相關(guān)聯(lián)。關(guān)于此項(xiàng)工作有兩種選擇。
          首 先,可使用filter-name和servlet-name子元素把此過(guò)濾器與一個(gè)特定的servlet名(此servlet名必須稍后在相同的 web.xml文件中使用servlet元素聲明)關(guān)聯(lián)。例如,下面的程序片斷指示系統(tǒng)只要利用一個(gè)定制的URL訪問(wèn)名為SomeServletName 的servlet或JSP頁(yè)面,就運(yùn)行名為Reporter的過(guò)濾器。
          <filter-mapping>
          <filter-name>Reporter</filter-name>
          <servlet-name>SomeServletName</servlet-name>
          </filter-mapping>
          其次,可利用filter-name和url-pattern子元素將過(guò)濾器與一組servlet、JSP頁(yè)面或靜態(tài)內(nèi)容相關(guān)聯(lián)。例如,相面的程序片段指示系統(tǒng)只要訪問(wèn)Web應(yīng)用中的任意URL,就運(yùn)行名為Reporter的過(guò)濾器。
          <filter-mapping>
          <filter-name>Reporter</filter-name>
          <url-pattern>/*</url-pattern>
          </filter-mapping>
          例 如,程序清單5-12給出了將ReportFilter過(guò)濾器與名為PageName的servlet相關(guān)聯(lián)的web.xml文件的一部分。名字 PageName依次又與一個(gè)名為T(mén)estPage.jsp的JSP頁(yè)面以及以模式http: //host/webAppPrefix/UrlTest2/ 開(kāi)頭的URL相關(guān)聯(lián)。TestPage.jsp的源代碼已經(jīng)JSP頁(yè)面命名的談?wù)撛谇懊娴?節(jié)"分配名稱(chēng)和定制的URL"中給出。事實(shí)上,程序清單5- 12中的servlet和servlet-name項(xiàng)從該節(jié)原封不動(dòng)地拿過(guò)來(lái)的。給定這些web.xml項(xiàng),可看到下面的標(biāo)準(zhǔn)輸出形式的調(diào)試報(bào)告(換行是 為了容易閱讀)。
          audit.irs.gov tried to access 
          http://mycompany.com/deployDemo/UrlTest2/business/tax-plan.html
          on Tue Dec 25 13:12:29 EDT 2001.

          程序清單5-12 Web.xml(說(shuō)明filter用法的摘錄)
          <?xml version="1.0" encoding="ISO-8859-1"?>
          <!DOCTYPE web-app
          PUBLIC "-//Sun Microsystems, Inc.//DTD Web Application 2.3//EN"
          "http://java.sun.com/dtd/web-app_2_3.dtd">

          <web-app>
          <filter>
          <filter-name>Reporter</filter-name>
          <filter-class>moresevlets.ReportFilter</filter-class>
          </filter>
          <!-- ... -->
          <filter-mapping>
          <filter-name>Reporter</filter-name>
          <servlet-name>PageName</servlet-name>
          </filter-mapping>
          <!-- ... -->
          <servlet>
          <servlet-name>PageName</servlet-name>
          <jsp-file>/RealPage.jsp</jsp-file>
          </servlet>
          <!-- ... -->
          <servlet-mapping>
          <servlet-name> PageName </servlet-name>
          <url-pattern>/UrlTest2/*</url-pattern>
          </servlet-mapping>
          <!-- ... -->
          </web-app>


          7 指定歡迎頁(yè)

          假 如用戶提供了一個(gè)像http: //host/webAppPrefix/directoryName/ 這樣的包含一個(gè)目錄名但沒(méi)有包含文件名的URL,會(huì)發(fā)生什么事情呢?用戶能得到一個(gè)目錄表?一個(gè)錯(cuò)誤?還是標(biāo)準(zhǔn)文件的內(nèi)容?如果得到標(biāo)準(zhǔn)文件內(nèi)容,是 index.html、index.jsp、default.html、default.htm或別的什么東西呢?
          Welcome-file-list 元素及其輔助的welcome-file元素解決了這個(gè)模糊的問(wèn)題。例如,下面的web.xml項(xiàng)指出,如果一個(gè)URL給出一個(gè)目錄名但未給出文件名,服 務(wù)器應(yīng)該首先試用index.jsp,然后再試用index.html。如果兩者都沒(méi)有找到,則結(jié)果有賴于所用的服務(wù)器(如一個(gè)目錄列表)。
          <welcome-file-list>
          <welcome-file>index.jsp</welcome-file>
          <welcome-file>index.html</welcome-file>
          </welcome-file-list>
          雖然許多服務(wù)器缺省遵循這種行為,但不一定必須這樣。因此,明確地使用welcom-file-list保證可移植性是一種良好的習(xí)慣。

          8 指定處理錯(cuò)誤的頁(yè)面

          現(xiàn) 在我了解到,你在開(kāi)發(fā)servlet和JSP頁(yè)面時(shí)從不會(huì)犯錯(cuò)誤,而且你的所有頁(yè)面是那樣的清晰,一般的程序員都不會(huì)被它們的搞糊涂。但是,是人總會(huì)犯錯(cuò) 誤的,用戶可能會(huì)提供不合規(guī)定的參數(shù),使用不正確的URL或者不能提供必需的表單字段值。除此之外,其它開(kāi)發(fā)人員可能不那么細(xì)心,他們應(yīng)該有些工具來(lái)克服 自己的不足。
          error-page元素就是用來(lái)克服這些問(wèn)題的。它有兩個(gè)可能的子元素,分別是:error-code和exception- type。第一個(gè)子元素error-code指出在給定的HTTP錯(cuò)誤代碼出現(xiàn)時(shí)使用的URL。第二個(gè)子元素excpetion-type指出在出現(xiàn)某個(gè) 給定的Java異常但未捕捉到時(shí)使用的URL。error-code和exception-type都利用location元素指出相應(yīng)的URL。此 URL必須以/開(kāi)始。location所指出的位置處的頁(yè)面可通過(guò)查找HttpServletRequest對(duì)象的兩個(gè)專(zhuān)門(mén)的屬性來(lái)訪問(wèn)關(guān)于錯(cuò)誤的信息, 這兩個(gè)屬性分別是:javax.servlet.error.status_code和javax.servlet.error.message。
          可回憶一下,在web.xml內(nèi)以正確的次序聲明web-app的子元素很重要。這里只要記住,error-page出現(xiàn)在web.xml文件的末尾附近,servlet、servlet-name和welcome-file-list之后即可。

          8.1 error-code元素
          為了更好地了解error-code元素的值,可考慮一下如果不正確地輸入文件名,大多數(shù)站點(diǎn)會(huì)作出什么反映。這樣做一般會(huì)出現(xiàn)一個(gè)404錯(cuò)誤信息,它表示不能找到該文件,但幾乎沒(méi)提供更多有用的信息。另一方面,可以試一下在www.microsoft.com、www.ibm.com 處或者特別是在www.bea.com 處輸出未知的文件名。這是會(huì)得出有用的消息,這些消息提供可選擇的位置,以便查找感興趣的頁(yè)面。提供這樣有用的錯(cuò)誤頁(yè)面對(duì)于Web應(yīng)用來(lái)說(shuō)是很有價(jià)值得。 事實(shí)上rm-error-page子元素)。由form-login-page給出的HTML表單必須具有一個(gè)j_security_check的 ACTION屬性、一個(gè)名為j_username的用戶名文本字段以及一個(gè)名為j_password的口令字段。
          例如,程序清單5-19指示服務(wù)器使用基于表單的驗(yàn)證。Web應(yīng)用的頂層目錄中的一個(gè)名為login.jsp的頁(yè)面將收集用戶名和口令,并且失敗的登陸將由相同目錄中名為login-error.jsp的頁(yè)面報(bào)告。

          程序清單5-19 web.xml(說(shuō)明login-config的摘錄)
          <?xml version="1.0" encoding="ISO-8859-1"?>
          <!DOCTYPE web-app
          PUBLIC "-//Sun Microsystems, Inc.//DTD Web Application 2.3//EN"
          "http://java.sun.com/dtd/web-app_2_3.dtd">

          <web-app>
          <!-- ... -->
          <security-constraint> ... </security-constraint>
          <login-config>
          <auth-method> FORM </auth-method>
          <form-login-config>
          <form-login-page>/login.jsp</form-login-page>
          <form-error-page>/login-error.jsp</form-error-page>
          </form-login-config>
          </login-config>
          <!-- ... -->
          </web-app>


          9.2 限制對(duì)Web資源的訪問(wèn)
          現(xiàn) 在,可以指示服務(wù)器使用何種驗(yàn)證方法了。"了不起,"你說(shuō)道,"除非我能指定一個(gè)來(lái)收到保護(hù)的 URL,否則沒(méi)有多大用處。"沒(méi)錯(cuò)。指出這些URL并說(shuō)明他們應(yīng)該得到何種保護(hù)正是security-constriaint元素的用途。此元素在 web.xml中應(yīng)該出現(xiàn)在login-config的緊前面。它包含是個(gè)可能的子元素,分別是:web-resource-collection、 auth-constraint、user-data-constraint和display-name。下面各小節(jié)對(duì)它們進(jìn)行介紹。
          l web-resource-collection
          此 元素確定應(yīng)該保護(hù)的資源。所有security-constraint元素都必須包含至少一個(gè)web-resource-collection項(xiàng)。此元素 由一個(gè)給出任意標(biāo)識(shí)名稱(chēng)的web-resource-name元素、一個(gè)確定應(yīng)該保護(hù)的URL的url-pattern元素、一個(gè)指出此保護(hù)所適用的 HTTP命令(GET、POST等,缺省為所有方法)的http-method元素和一個(gè)提供資料的可選description元素組成。例如,下面的 Web-resource-collection項(xiàng)(在security-constratint元素內(nèi))指出Web應(yīng)用的proprietary目錄中 所有文檔應(yīng)該受到保護(hù)。
          <security-constraint>
          <web-resource-coolection>
          <web-resource-name>Proprietary</web-resource-name>
          <url-pattern>/propritary/*</url-pattern>
          </web-resource-coolection>
          <!-- ... -->
          </security-constraint>
          重 要的是應(yīng)該注意到,url-pattern僅適用于直接訪問(wèn)這些資源的客戶機(jī)。特別是,它不適合于通過(guò)MVC體系結(jié)構(gòu)利用 RequestDispatcher來(lái)訪問(wèn)的頁(yè)面,或者不適合于利用類(lèi)似jsp:forward的手段來(lái)訪問(wèn)的頁(yè)面。這種不勻稱(chēng)如果利用得當(dāng)?shù)脑捄苡泻?nbsp;處。例如,servlet可利用MVC體系結(jié)構(gòu)查找數(shù)據(jù),把它放到bean中,發(fā)送請(qǐng)求到從bean中提取數(shù)據(jù)的JSP頁(yè)面并顯示它。我們希望保證決不直 接訪問(wèn)受保護(hù)的JSP頁(yè)面,而只是通過(guò)建立該頁(yè)面將使用的bean的servlet來(lái)訪問(wèn)它。url-pattern和auth-contraint元素 可通過(guò)聲明不允許任何用戶直接訪問(wèn)JSP頁(yè)面來(lái)提供這種保證。但是,這種不勻稱(chēng)的行為可能讓開(kāi)發(fā)人員放松警惕,使他們偶然對(duì)應(yīng)受保護(hù)的資源提供不受限制的 訪問(wèn)。 
          l auth-constraint
          盡管web-resource-collention元素質(zhì)出了哪些URL應(yīng)該受到保護(hù), 但是auth-constraint元素卻指出哪些用戶應(yīng)該具有受保護(hù)資源的訪問(wèn)權(quán)。此元素應(yīng)該包含一個(gè)或多個(gè)標(biāo)識(shí)具有訪問(wèn)權(quán)限的用戶類(lèi)別role- name元素,以及包含(可選)一個(gè)描述角色的description元素。例如,下面web.xml中的security-constraint元素部 門(mén)規(guī)定只有指定為Administrator或Big Kahuna(或兩者)的用戶具有指定資源的訪問(wèn)權(quán)。
          <security-constraint>
          <web-resource-coolection> ... </web-resource-coolection>
          <auth-constraint>
          <role-name>administrator</role-name>
          <role-name>kahuna</role-name>
          </auth-constraint>
          </security-constraint>
          重要的是認(rèn)識(shí)到,到此為止,這個(gè)過(guò)程的可移植部分結(jié)束了。服務(wù)器怎樣確定哪些用戶處于任何角色以及它怎樣存放用戶的口令,完全有賴于具體的系統(tǒng)。
          例如,Tomcat使用install_dir/conf/tomcat-users.xml將用戶名與角色名和口令相關(guān)聯(lián),正如下面例子中所示,它指出用戶joe(口令bigshot)和jane(口令enaj)屬于administrator和kahuna角色。
          <tomcat-users>
          <user name="joe" password="bigshot" roles="administrator,kahuna" />
          <user name="jane" password="enaj" roles="kahuna" />
          </tomcat-users>
          l user-data-constraint
          這 個(gè)可選的元素指出在訪問(wèn)相關(guān)資源時(shí)使用任何傳輸層保護(hù)。它必須包含一個(gè)transport-guarantee子元素(合法值為NONE、 INTEGRAL或CONFIDENTIAL),并且可選地包含一個(gè)description元素。transport-guarantee為NONE值將 對(duì)所用的通訊協(xié)議不加限制。INTEGRAL值表示數(shù)據(jù)必須以一種防止截取它的人閱讀它的方式傳送。雖然原理上(并且在未來(lái)的HTTP版本中),在 INTEGRAL和CONFIDENTIAL之間可能會(huì)有差別,但在當(dāng)前實(shí)踐中,他們都只是簡(jiǎn)單地要求用SSL。例如,下面指示服務(wù)器只允許對(duì)相關(guān)資源做 HTTPS連接:
          <security-constraint>
          <!-- ... -->
          <user-data-constraint>
          <transport-guarantee>CONFIDENTIAL</transport-guarantee>
          </user-data-constraint>
          </security-constraint>
          l display-name
          security-constraint的這個(gè)很少使用的子元素給予可能由GUI工具使用的安全約束項(xiàng)一個(gè)名稱(chēng)。
          9.3 分配角色名
          迄今為止,討論已經(jīng)集中到完全由容器(服務(wù)器)處理的安全問(wèn)題之上了。但servlet以及JSP頁(yè)面也能夠處理它們自己的安全問(wèn)題。
          例 如,容器可能允許用戶從bigwig或bigcheese角色訪問(wèn)一個(gè)顯示主管人員額外緊貼的頁(yè)面,但只允許bigwig用戶修改此頁(yè)面的參數(shù)。完成這種 更細(xì)致的控制的一種常見(jiàn)方法是調(diào)用HttpServletRequset的isUserInRole方法,并據(jù)此修改訪問(wèn)。
          Servlet的 security-role-ref子元素提供出現(xiàn)在服務(wù)器專(zhuān)用口令文件中的安全角色名的一個(gè)別名。例如,假如編寫(xiě)了一個(gè)調(diào)用 request.isUserInRole("boss")的servlet,但后來(lái)該servlet被用在了一個(gè)其口令文件調(diào)用角色manager而不 是boss的服務(wù)器中。下面的程序段使該servlet能夠使用這兩個(gè)名稱(chēng)中的任何一個(gè)。
          <servlet>
          <!-- ... -->
          <security-role-ref>
          <role-name>boss</role-name> <!-- New alias -->
          <role-link>manager</role-link> <!-- Real name -->
          </security-role-ref>
          </servlet>
          也可以在web-app內(nèi)利用security-role元素提供將出現(xiàn)在role-name元素中的所有安全角色的一個(gè)全局列表。分別地生命角色使高級(jí)IDE容易處理安全信息。

          10 控制會(huì)話超時(shí)

          如 果某個(gè)會(huì)話在一定的時(shí)間內(nèi)未被訪問(wèn),服務(wù)器可把它扔掉以節(jié)約內(nèi)存。可利用HttpSession的setMaxInactiveInterval方法直接 設(shè)置個(gè)別會(huì)話對(duì)象的超時(shí)值。如果不采用這種方法,則缺省的超時(shí)值由具體的服務(wù)器決定。但可利用session-config和session- timeout元素來(lái)給出一個(gè)適用于所有服務(wù)器的明確的超時(shí)值。超時(shí)值的單位為分鐘,因此,下面的例子設(shè)置缺省會(huì)話超時(shí)值為三個(gè)小時(shí)(180分鐘)。
          <session-config>
          <session-timeout>180</session-timeout>
          </session-config>

          11 Web應(yīng)用的文檔化

          越 來(lái)越多的開(kāi)發(fā)環(huán)境開(kāi)始提供servlet和JSP的直接支持。例子有Borland Jbuilder Enterprise Edition、Macromedia UltraDev、Allaire JRun Studio(寫(xiě)此文時(shí),已被Macromedia收購(gòu))以及IBM VisuaAge for Java等。
          大量的web.xml元素不僅是為服務(wù)器設(shè)計(jì)的,而且還是為可視開(kāi)發(fā)環(huán)境設(shè)計(jì)的。它們包括icon、display-name和discription等。
          可回憶一下,在web.xml內(nèi)以適當(dāng)?shù)卮涡蚵暶鱳eb-app子元素很重要。不過(guò),這里只要記住icon、display-name和description是web.xml的web-app元素內(nèi)的前三個(gè)合法元素即可。
          l icon
          icon元素指出GUI工具可用來(lái)代表Web應(yīng)用的一個(gè)和兩個(gè)圖像文件。可利用small-icon元素指定一幅16 x 16的GIF或JPEG圖像,用large-icon元素指定一幅32 x 32的圖像。下面舉一個(gè)例子: 
          <icon>
          <small-icon>/images/small-book.gif</small-icon>
          <large-icon>/images/tome.jpg</large-icon>
          </icon>
          l display-name
          display-name元素提供GUI工具可能會(huì)用來(lái)標(biāo)記此Web應(yīng)用的一個(gè)名稱(chēng)。下面是個(gè)例子。
          <display-name>Rare Books</display-name>
          l description
          description元素提供解釋性文本,如下所示:
          <description>
          This Web application represents the store developed for
          rare-books.com, an online bookstore specializing in rare
          and limited-edition books.
          </description>
          posted @ 2007-07-02 13:05 和田雨 閱讀(220) | 評(píng)論 (0)編輯 收藏

          1. 如何讓選中的一行記錄高亮顯示?
          http://topic.csdn.net/t/20050105/09/3699823.html
          2. 如何在下拉列表框里選擇一個(gè)值后跳出新窗口?
          http://topic.csdn.net/t/20050107/15/3707444.html
          3. 如何在JSP中啟動(dòng)execl?
          http://community.csdn.net/Expert/TopicView1.asp?id=4909154
          http://community.csdn.net/Expert/TopicView1.asp?id=4685297
          4. 兩級(jí)聯(lián)動(dòng)菜單
          http://topic.csdn.net/t/20040730/16/3227437.html
          5. java中如何把一個(gè)目錄下的文件移到另一個(gè)指定的目錄?
          http://topic.csdn.net/t/20050112/13/3719237.html
          6. 如何制作表格線?
          http://community.csdn.net/Expert/TopicView1.asp?id=5130862
          7. jsp如判別一個(gè)字符在A到Z之間?
          http://topic.csdn.net/t/20050113/08/3721286.html
          8. 得到一浮點(diǎn)數(shù)小數(shù)點(diǎn)后4位,如何寫(xiě)以函數(shù)截取為兩位
          http://topic.csdn.net/t/20050117/14/3731712.html
          9. 整型轉(zhuǎn)字符?
          http://topic.csdn.net/t/20030526/22/1837632.html
          10. 顯示數(shù)據(jù)庫(kù)的記錄,點(diǎn)擊哪個(gè)標(biāo)題欄,就按照哪個(gè)標(biāo)題排序
          http://topic.csdn.net/t/20041226/15/3677011.html
          11. 制作兩個(gè)按鈕
          ??
          12. 用什么sql語(yǔ)句將數(shù)據(jù)庫(kù)中的一條記錄,復(fù)制為兩條?
          http://topic.csdn.net/t/20050120/17/3741746.html
          13. 有兩個(gè)按鈕A,B,其中A按下去,A就不可以再按,另外B按下去,A可以再按..請(qǐng)問(wèn)如何用javascript控制.?
          http://topic.csdn.net/t/20050120/17/3741568.html
          14. 請(qǐng)問(wèn)jsp中的數(shù)據(jù)庫(kù)的連接方式
          http://topic.csdn.net/t/20030107/11/1334377.html
          15. 在jsp中,怎么實(shí)現(xiàn)按回車(chē)就可提交表單?
          http://topic.csdn.net/t/20050121/11/3743133.html
          16. 在JSP中如何傳遞數(shù)組?
          http://topic.csdn.net/t/20030624/17/1952227.html
          17. 如何按地址取圖片?
          http://community.csdn.net/Expert/TopicView1.asp?id=4894525
          http://community.csdn.net/Expert/TopicView1.asp?id=4990451
          18. JSP中如何上傳圖片到數(shù)據(jù)庫(kù)字段?
          http://community.csdn.net/Expert/TopicView1.asp?id=4923127
          19. 頁(yè)面自動(dòng)刷新?
          http://topic.csdn.net/t/20040430/16/3025697.html
          20. 表單自動(dòng)提交?
          http://community.csdn.net/Expert/TopicView1.asp?id=4681334
          21. 如何從JSP傳數(shù)據(jù)的到JAVABEAN里?
          這個(gè)不就是參數(shù)傳遞么??
          22. weblogic 數(shù)據(jù)池連接? 數(shù)據(jù)源 JNDI名字 mysource
          http://topic.csdn.net/t/20040915/22/3376617.html
          23. 三級(jí)聯(lián)動(dòng)菜單?
          http://blog.csdn.net/overmind/archive/2005/10/27/517856.aspx
          24. 在JSP中如何調(diào)用瀏覽器中的"另存為"功能?
          http://topic.csdn.net/t/20050509/09/3990931.html
          25. 網(wǎng)頁(yè)全屏顯示
          http://community.csdn.net/Expert/TopicView1.asp?id=5057245
          26. 求兩個(gè)日期相隔了多少天:輸入時(shí)間格式為(yyyy-mm-dd)
          http://topic.csdn.net/t/20050516/10/4009546.html
          27. 上傳文件對(duì)話框
          http://community.csdn.net/Expert/TopicView1.asp?id=4961980
          28. 分頁(yè)測(cè)試
          http://community.csdn.net/Expert/TopicView1.asp?id=4925123
          29. 下載文件
          http://community.csdn.net/Expert/TopicView1.asp?id=5019146
          30. 簡(jiǎn)單漢字判斷
          http://blog.csdn.net/duanxd/archive/2005/05/19/376481.aspx
          31. 可否將JSP的變量或者bean傳遞給JAVASCRIPT使用?
          http://topic.csdn.net/t/20050520/14/4022667.html
          32. 怎么把一種格式(如yyyy-mm-dd)的日期變量轉(zhuǎn)換成另外一種格式(如dd-mm-yyyy)的日期變量!
          http://topic.csdn.net/t/20050602/21/4055209.html
          33. 如何實(shí)現(xiàn)在頁(yè)面上添加一行輸入對(duì)話框
          ??
          34. 如何制作驗(yàn)證碼圖片?
          http://community.csdn.net/Expert/TopicView1.asp?id=4857455
          35. 圖片縮小放大功能
          http://topic.csdn.net/t/20041227/09/3678138.html
          36. 如何點(diǎn)擊輸入框彈出日期選擇?
          http://community.csdn.net/Expert/TopicView1.asp?id=5020494
          http://www.dynarch.com/projects/calendar/
          37. 如何在圖片上傳前查看自己要上傳的圖片?
          http://community.csdn.net/Expert/TopicView1.asp?id=4825282
          38. 如何在圖片排列的頁(yè)面上選中一個(gè)圖片,然后把它刪掉?
          http://community.csdn.net/Expert/TopicView1.asp?id=5162107
          39. 顯示某目錄下的所有文件
          http://community.csdn.net/Expert/TopicView1.asp?id=5056503
          40. 如何把復(fù)選框中選中的值直接顯示在輸入文本框中?
          http://community.csdn.net/Expert/TopicView1.asp?id=5004830
          41. 如何選擇路徑下的文件直接顯示文件內(nèi)容?
          http://community.csdn.net/Expert/TopicView1.asp?id=5194181
          http://community.csdn.net/Expert/TopicView1.asp?id=5121413
          42. 如何制作進(jìn)度條?
          http://topic.csdn.net/t/20051120/03/4405364.html
          43. 如何對(duì)較長(zhǎng)字符省略顯示?
          http://community.csdn.net/Expert/TopicView1.asp?id=5124078
          44. 如何制作日歷?
          這個(gè)太多了
          45. 點(diǎn)擊文本彈出一個(gè)選擇框,選中值后返回并顯示
          http://community.csdn.net/Expert/TopicView1.asp?id=4979197
          http://community.csdn.net/Expert/TopicView1.asp?id=4988006
          46. 如何關(guān)閉框架頁(yè)面回到?jīng)]框架的頁(yè)面?
          http://community.csdn.net/Expert/TopicView1.asp?id=4970057
          47. 如何選中多條記錄一次性刪除?
          http://community.csdn.net/Expert/TopicView1.asp?id=4762839
          48. 如何自動(dòng)獲取頁(yè)面文件名?
          http://community.csdn.net/Expert/TopicView1.asp?id=4945500
          49. 如何在JSP中調(diào)用ActiveX控件?
          http://community.csdn.net/Expert/TopicView1.asp?id=4919689
          50. 如何使用SmartUpload實(shí)現(xiàn)文件上傳?
          http://community.csdn.net/Expert/TopicView1.asp?id=4911791
          51. 如何使用iReport和Jasperreport開(kāi)發(fā)報(bào)表?
          http://community.csdn.net/Expert/TopicView1.asp?id=5038328
          52. 如何使用iText生成PDF?
          http://www.lowagie.com/iText/tutorial/ch07.html#htmlparsing
          http://java.ccidnet.com/art/3565/20050309/480327_1.html
          53. 如何制作圖片水印?
          http://community.csdn.net/Expert/TopicView1.asp?id=5212770
          http://community.csdn.net/Expert/TopicView1.asp?id=4885889
          54. 如何在頁(yè)面中屏蔽鍵盤(pán)功能鍵?
          http://community.csdn.net/Expert/TopicView1.asp?id=4356360
          55. 如何禁止用戶復(fù)制網(wǎng)頁(yè)內(nèi)容?
          http://community.csdn.net/Expert/TopicView1.asp?id=4356360
          56. 如何實(shí)現(xiàn)不在地址攔中顯示當(dāng)前URL?
          http://community.csdn.net/Expert/TopicView1.asp?id=4678351
          http://community.csdn.net/Expert/TopicView1.asp?id=4863808
          57. 如何獲取用戶的真實(shí)IP地址?
          http://community.csdn.net/Expert/TopicView1.asp?id=4718686
          58. 如何獲取用戶瀏覽器信息?
          http://community.csdn.net/Expert/TopicView1.asp?id=4987045
          http://community.csdn.net/Expert/TopicView1.asp?id=4898974
          59. 如何獲取當(dāng)前絕對(duì)路徑?
          http://community.csdn.net/Expert/TopicView1.asp?id=5157921
          http://community.csdn.net/Expert/TopicView1.asp?id=5178848
          60. 如何將HTML文件轉(zhuǎn)換成XML文件?
          http://community.csdn.net/Expert/TopicView1.asp?id=4542984
          61. 如何結(jié)合XML和XSL輸出HTML頁(yè)面?
          http://community.csdn.net/Expert/TopicView1.asp?id=4355891
          62. 如何制作動(dòng)態(tài)樹(shù)型菜單制作?
          http://www.aygfsteel.com/rickhunter/articles/59742.html
          http://community.csdn.net/Expert/TopicView1.asp?id=4966014
          63. 如何制作類(lèi)似QQ的短消息提示?
          http://community.csdn.net/Expert/TopicView1.asp?id=4702037
          http://community.csdn.net/Expert/TopicView1.asp?id=5087584
          64. 如何用jfreechat制作拄、餅、曲線圖型?
          http://www.inspiresky.com/Article/ShowArticle.asp?ArticleID=731
          65. 文本框怎么只容許輸入數(shù)字?
          http://community.csdn.net/Expert/TopicView1.asp?id=4927727
          66. 如何打開(kāi)Word和Execl文件?
          http://community.csdn.net/Expert/TopicView1.asp?id=5194181
          http://community.csdn.net/Expert/TopicView1.asp?id=4685297
          67. 如何生成Word 和 Excel文檔?
          http://sourceforge.net/project/showfiles.php?group_id=15255&release_id=167948
          68. JSP如何讀取Word內(nèi)容?
          http://topic.csdn.net/t/20030217/13/1435281.html 
          posted @ 2007-07-02 13:01 和田雨 閱讀(558) | 評(píng)論 (0)編輯 收藏

           1、獲取服務(wù)器端當(dāng)前日期:
          <%@ page import="java.util.Date"%>
          <%
          Date myDate = new Date();
          %>
          2、獲取當(dāng)前年、月、日:

          <%@ page import="java.util.Date"%>

          <%
          Date myDate = new Date();
          int thisYear = myDate.getYear() + 1900;//thisYear = 2003
          int thisMonth = myDate.getMonth() + 1;//thisMonth = 5
          int thisDate = myDate.getDate();//thisDate = 30
          %>


          3、按本地時(shí)區(qū)輸出當(dāng)前日期

          <%@ page import="java.util.Date"%>
          <%
          Date myDate = new Date();
          out.println(myDate.toLocaleString());
          %>


          輸出結(jié)果為:
          2003-5-30


          4、獲取數(shù)據(jù)庫(kù)中字段名為"publish_time"、類(lèi)型為Datetime的值


          <%@ page import="java.util.Date"%>
          <%
          ...連接數(shù)據(jù)庫(kù)...
          ResultSet rs = ...
          Date sDate = rs.getDate("publish_time");
          %>
          [code]


          5、按照指定格式打印日期


          [code]
          <%@ page import="java.util.Date"%>
          <%@ page import="java.text.DateFormat"%>
          <%
          Date dNow = new Date();

          SimpleDateFormat formatter = new SimpleDateFormat("E yyyy.MM.dd 'at' hh:mm:ss a zzz");
          out.println("It is " + formatter.format(dNow));
          %>

           

          輸出的結(jié)果為:
          It is 星期五 2003.05.30 at 11:30:46 上午 CST
          (更為詳盡的格式符號(hào)請(qǐng)參看SimpleDateFormat類(lèi))


          6、將字符串轉(zhuǎn)換為日期

          <%@ page import="java.util.Date"%>
          <%@ page import="java.text.DateFormat"%>
          <%
          String input = "1222-11-11";
          SimpleDateFormat formatter = new SimpleDateFormat("yyyy-MM-dd");
          Date t = null;

          try{

            t = formatter.parse(input);

            out.println(t);

            }catch(ParseException e){

            out.println("unparseable using" + formatter);

            }

          %>

          輸出結(jié)果為:
          Fri Nov 11 00:00:00 CST 1222


          7、計(jì)算日期之間的間隔

           

          <%@ page import="java.util.Date"%>
          <%@ page import="java.text.DateFormat"%>
          <%
          String input = "2003-05-01";
          SimpleDateFormat formatter = new SimpleDateFormat("yyyy-MM-dd");
          Date d1 = null;
          trycatch(ParseException e)

          Date d2 = new Date();

          long diff = d2.getTime() - d1.getTime();
          out.println("Difference is " + (diff/(1000*60*60*24)) + " days.");
          %>


          輸出結(jié)果為:
          Difference is 29 days.


          8、日期的加減運(yùn)算
          方法:用Calendar類(lèi)的add()方法


          <%@ page import="java.util.*"%>
          <%@ page import="java.text.*"%>
          <%
          Calendar now = Calendar.getInstance();
          SimpleDateFormat formatter = new SimpleDateFormat("E yyyy.MM.dd 'at' hh:mm:ss a zzz");
          out.println("It is now " + formatter.format(now.getTime()));
          now.add(Calendar.DAY_OF_YEAR,-(365*2));
          out.println("<br>");
          out.println("Two years ago was " + formatter.format(now.getTime()));
          %>

           

          輸出結(jié)果為:
          It is now 星期五 2003.05.30 at 01:45:32 下午 CST
          Two years ago was 星期三 2001.05.30 at 01:45:32 下午 CST


          9、比較日期
          方法:用equals()、before()、after()方法

           

          <%@ page import="java.util.*"%>
          <%@ page import="java.text.*"%>
          <%
          DateFormat df = new SimpleDateFormat("yyy-MM-dd");
          Date d1 = df.parse("2000-01-01");
          Date d2 = df.parse("1999-12-31");

          String relation = null;
          if(d1.equals(d2))
          relation = "the same date as";
          else if(d1.before(d2))
          relation = "before";
          else
          relation = "after";
          out.println(d1 +" is " + relation + ' ' + d2);
          %>


          輸出結(jié)果為:
          Sat Jan 01 00:00:00 CST 2000 is after Fri Dec 31 00:00:00 CST 1999


          10、記錄一件事所花費(fèi)的時(shí)間
          方法:調(diào)用兩次System.getTimeMillis()方法,求差值

           

          <%@ page import="java.text.*"%>
          <%
          long t0,t1;
          t0 = System.currentTimeMillis();
          out.println("Cyc starts at " + t0);
          int k = 0;
          for(int i =0;i<100000;i++)
          t1 = System.currentTimeMillis();
          out.println("<br>");
          out.println("Cyc ends at " + t1);
          out.println("<br>");
          out.println("This run took " + (t1-t0) + "ms.");
          %>


          輸出結(jié)果為:
          Cyc starts at 1054275312432
          Cyc ends at 1054275312442
          This run took 10ms.

          其它:如何格式化小數(shù)

           

          <%@ page import="java.text.*"%>
          <%
          DecimalFormat df = new DecimalFormat(",###.00");
          double aNumber = 33665448856.6568975;
          String result = df.format(aNumber);
          out.println(result);
          %>


          輸出結(jié)果為:
          33,665,448,856.66

          posted @ 2007-07-02 13:01 和田雨 閱讀(230) | 評(píng)論 (0)編輯 收藏

          計(jì)數(shù)器是一般網(wǎng)站必備的東東,別小看它了,每當(dāng)站長(zhǎng)看著小小計(jì)數(shù)器上的數(shù)字飛速增長(zhǎng)的時(shí)候,感覺(jué)實(shí)在是好極了。以前我們用cgi、asp來(lái)寫(xiě)計(jì)數(shù)器,這方面的文章很多了,在這里,我們將會(huì)采用目前比較流行的jsp技術(shù)演示如何做一個(gè)計(jì)數(shù)器。 6?P~(MPPm  
          mVfGL 3P  
            其中我們用到了兩個(gè)文件,test.jsp文件用于在瀏覽器中運(yùn)行,counter.java是后臺(tái)的一個(gè)小java bean程序,用來(lái)讀計(jì)數(shù)器的值和寫(xiě)入計(jì)數(shù)器的值。而對(duì)于計(jì)數(shù)器的保存,我們采用了一個(gè)文本文件lyfcount.txt。
          B2cYy,Whi  
          6Nz</bHn  
          轉(zhuǎn)自:動(dòng)態(tài)網(wǎng)制作指南
          www.knowsky.com '$ 6Zv$S  
          mgpeMc5R3  
          下面是詳細(xì)的程序代碼(test.jsp放到web目錄下,counter.java放到class目錄):
          +33qS}  
          //test.jsp文件
          @`3-X/Q  
          <%@ page contentType="text/html;charset=gb2312"%>
          LYL. eK  
          <HTML>
           pW0  
          <HEAD>
          C|F1F &2U  
          <meta http-equiv="Content-Type" content="text/html; charset=gb2312">
          &wk.'0R9E  
          <META NAME="GENERATOR" CONTENT="Oracle JDeveloper">
          %K&gZIkX  
          <TITLE>
          87x7Rnj >=  
          計(jì)數(shù)器演示程序
          %fU wb  
          </TITLE>
          "I,z,/5Y  
          </HEAD>
          lJgBcZ{7  
          <BODY>
          IrtugFf  
          <!--創(chuàng)建并調(diào)用bean(counter)-->
          KPX)E!"v  
          <jsp:useBean id="counter" class="counter" scope="request">
          v2LW[I7]  
          </jsp:useBean>
          -5~k  
          <%
          ~jKx  
          //調(diào)用counter對(duì)象的ReadFile方法來(lái)讀取文件lyfcount.txt中的計(jì)數(shù)
          j(}8B=c{e6  
          String cont=counter.ReadFile("/lyfcount.txt");
          s Ki/puJ  
          //調(diào)用counter對(duì)象的ReadFile方法來(lái)將計(jì)數(shù)器加一后寫(xiě)入到文件lyfcount.txt中
          b#j^QB9w N  
          counter.WriteFile("/lyfcount.txt",cont);%>
          W*AWh`yDK  
          您是第<font color="red"><%=cont%></font>位訪問(wèn)者
          j {i6 TI  
          </BODY>
          e$]VJ-*Ni  
          </HTML>
          ,x 61Tj6p  
          {enOdTJQx  
          //counter.java 讀寫(xiě)文件的一個(gè)bean
          }-0Rke6`>Y  
          import java.io.*;
          pcj}E(\ y  
          mc?=kzE  
          public class counter extends Object {
          NHF1l}*  
          private String currentRecord = null;//保存文本的變量
          0&dB8+;Ge  
          private BufferedReader file; //BufferedReader對(duì)象,用于讀取文件數(shù)據(jù)
          Pv: H87  
          private String path;//文件完整路徑名
           B4&% )2  
          public counter() {
          ] RgX{;  
          }
          h4Hi&30  
          //ReadFile方法用來(lái)讀取文件filePath中的數(shù)據(jù),并返回這個(gè)數(shù)據(jù)
          ZF%0a,&%  
          public String ReadFile(String filePath) throws FileNotFoundException
          _v(ojt r=  
          {
          1HPnp /u  
          path = filePath;
          YicVk\(1  
          //創(chuàng)建新的BufferedReader對(duì)象
          2~ 0K7~  
          file = new BufferedReader(new FileReader(path));
          ct3vuSf?  
          String returnStr =null;
          &vt{]z=  
          try
          <ymk1tk8@  
          {
          p|~K$ +R  
          //讀取一行數(shù)據(jù)并保存到currentRecord變量中
          ~-g#'  
          currentRecord = file.readLine();
          C{s%WW3?  
          }
          ! f/|Lm  
          catch (IOException e)
          DEC5"S'f#  
          {//錯(cuò)誤處理
          j^ %9ji  
          System.out.println("讀取數(shù)據(jù)錯(cuò)誤.");
          uiro2 Fz8  
          }
          }H&/l?g'  
          if (currentRecord == null)
          DgO0/n  
          //如果文件為空
          Y:d2ra*[&  
          returnStr = "沒(méi)有任何記錄";
          `Au)H  
          else
          FN@p~ X  
          {//文件不為空
          ="p{vnN"  
          returnStr =currentRecord;
          ]v GW  
          }
          QDanl36  
          //返回讀取文件的數(shù)據(jù)
          AR@urdG  
          return returnStr;
          V0 @M$op  
          }
          e i&:Z  
          //ReadFile方法用來(lái)將數(shù)據(jù)counter+1后寫(xiě)入到文本文件filePath中
          c$h=gQ  
          //以實(shí)現(xiàn)計(jì)數(shù)增長(zhǎng)的功能
          ~tnQw}x  
          public void WriteFile(String filePath,String counter) throws FileNotFoundException
          afTiP0 Qz  
          {
          dnx4bUW0  
          path = filePath;
           *% Ga  
          //將counter轉(zhuǎn)換為int類(lèi)型并加一
          ! Al9>  
          int Writestr = Integer.parseInt(counter)+1;
          ^$%vQ/F,Z  
          try {
          C{A%JXV|y  
          //創(chuàng)建PrintWriter對(duì)象,用于寫(xiě)入數(shù)據(jù)到文件中
          oFhww4mY|  
          PrintWriter pw = new PrintWriter(new FileOutputStream(filePath));
          QyjT?K{j:  
          //用文本格式打印整數(shù)Writestr
          *R\U6A=J8  
          pw.println(Writestr);
          S$C5 c .  
          //清除PrintWriter對(duì)象
          >|=}0K  
          pw.close();
          AsBtn-X>^  
          } catch(IOException e) {
          (1XAc\Le  
          //錯(cuò)誤處理
          e_Pdm8Lw)  
          System.out.println("寫(xiě)入文件錯(cuò)誤"+e.getMessage());
          5._Wa[@  
          }
          25A-F<x  
          }
          BnUr~  
          _f A:ynT]{  
          }
          #i'a`8Y6QT  
          U_mAGr[  
            到這里,程序?qū)懲炅耍瑢ounter.java編譯為counter.class,同樣放在對(duì)應(yīng)的class目錄下,在根目錄下建立一個(gè)lyfcount.txt文件,文件內(nèi)容就一個(gè)數(shù)字0,直接在瀏覽器中敲入地址就可以看到計(jì)數(shù)器了,刷新瀏覽器會(huì)看到不斷變幻的數(shù)字。
          mCzk^}5  
          @(;G`0C  
            (如果運(yùn)行時(shí)候提示找不到文件,請(qǐng)將上面test.jsp中的readfile那一句注釋后運(yùn)行一次則lyfcount.txt文件自動(dòng)建立,然后就可以正常運(yùn)行。)

          posted @ 2007-07-02 12:59 和田雨 閱讀(215) | 評(píng)論 (0)編輯 收藏

          目前, 國(guó)內(nèi)外信息化建設(shè)已經(jīng)進(jìn)入基于Web應(yīng)用為核心的階段, Java作為應(yīng)用于網(wǎng)絡(luò)的最好語(yǔ)言,前景無(wú)限看好。然而,就算用Java建造一個(gè)不是很煩瑣的web應(yīng)用,也不是件輕松的事情。概括一下,實(shí)施Java的WEB項(xiàng)目需要掌握的技術(shù)如下:

           
          lJava語(yǔ)言
          l面向?qū)ο蠓治鲈O(shè)計(jì)思想
          l設(shè)計(jì)模式和框架結(jié)構(gòu)
          lXML語(yǔ)言
          l網(wǎng)頁(yè)腳本語(yǔ)言
          l數(shù)據(jù)庫(kù)
          l應(yīng)用服務(wù)器
          l集成開(kāi)發(fā)環(huán)境
           
          下面我們具體地看每個(gè)技術(shù).
          1、Java語(yǔ)言

          Java
          語(yǔ)言體系比較龐大,包括多個(gè)模塊。從WEB項(xiàng)目應(yīng)用角度講有JSP、Servlet、JDBC、JavaBean(Application)四部分技術(shù)。
           
          (1)、Java Database Connectivity (JDBC)技術(shù)
          Java Web應(yīng)用開(kāi)發(fā)中,數(shù)據(jù)庫(kù)管理系統(tǒng)(RDBMS)的使用是不可缺少的。JDBC(Java Database Connectivity) 是一種用于執(zhí)行 SQL 語(yǔ)句的 Java API。它由一組用 Java 編程語(yǔ)言編寫(xiě)的類(lèi)和接口組成。JDBC 為工具/數(shù)據(jù)庫(kù)開(kāi)發(fā)人員提供了一個(gè)標(biāo)準(zhǔn)的API,使他們能夠用純Java API 來(lái)編寫(xiě)數(shù)據(jù)庫(kù)應(yīng)用程序。
          簡(jiǎn)單地說(shuō),JDBC 可做三件事:
          l與數(shù)據(jù)庫(kù)建立連接,
          l發(fā)送 SQL 語(yǔ)句,
          l處理結(jié)果。
           
           (2)、Servlet技術(shù)
          Servlet是運(yùn)行在服務(wù)器端的程序,可以被認(rèn)為是服務(wù)器端的applet。servlet被Web服務(wù)器(例如Tomcat)加載和執(zhí)行,就如同applet被瀏覽器加載和執(zhí)行一樣。servlet從客戶端(通過(guò)Web服務(wù)器)接收請(qǐng)求,執(zhí)行某種操作,然后返回結(jié)果。
           
          Servlet的主要優(yōu)點(diǎn)包括
          lServlet是持久的。servlet只需Web服務(wù)器加載一次,而且可以在不同請(qǐng)求之間保持服務(wù)(例如一次數(shù)據(jù)庫(kù)連接)。
          lServlet是與平臺(tái)無(wú)關(guān)的。如前所述,servlet是用Java編寫(xiě)的,它自然也繼承了Java的平臺(tái)無(wú)關(guān)性。
          lServlet是可擴(kuò)展的。由于servlet是用Java編寫(xiě)的,它就具備了Java所能帶來(lái)的所有優(yōu)點(diǎn)。Java是健壯的、面向?qū)ο蟮木幊陶Z(yǔ)言,它很容易擴(kuò)展以適應(yīng)你的需求。servlet自然也具備了這些特征。
          lServlet是安全的。從外界調(diào)用一個(gè)servlet的惟一方法就是通過(guò)Web服務(wù)器。這提供了高水平的安全性保障,尤其是在你的Web服務(wù)器有防火墻保護(hù)的時(shí)候。
          lServlet可以在多種多樣的客戶機(jī)上使用。由于servlet是用Java編寫(xiě)的,所以你可以很方便地在HTML中使用它們。
           
          (3)、JavaServer Pages(JSP) 技術(shù)
          JSP是從Servlet上分離出來(lái)的一小部分,簡(jiǎn)化了開(kāi)發(fā),加強(qiáng)了界面設(shè)計(jì)。JSP定位在交互網(wǎng)頁(yè)的開(kāi)發(fā)。運(yùn)用Java語(yǔ)法,但功能較Servlet弱了很多,并且高級(jí)開(kāi)發(fā)中只充當(dāng)用戶界面部分。JSP容器收到客戶端發(fā)出的請(qǐng)求時(shí),首先執(zhí)行其中的程序片段,然后將執(zhí)行結(jié)果以HTML格式響應(yīng)給客戶端。其中程序片段可以是:操作數(shù)據(jù)庫(kù)、重新定向網(wǎng)頁(yè)以及發(fā)送 E-Mail 等等,這些都是建立動(dòng)態(tài)網(wǎng)站所需要的功能。所有程序操作都在服務(wù)器端執(zhí)行,網(wǎng)絡(luò)上傳送給客戶端的僅是得到的結(jié)果,與客戶端的瀏覽器無(wú)關(guān),因此,JSP 稱(chēng)為Server-Side Language。
           
          JavaServer Pages的主要優(yōu)點(diǎn)包括
          ●一次編寫(xiě),各處執(zhí)行(Write o­nce, Run Anywhere)特性
          作為Java 平臺(tái)的一部分,JavaServer Pages 技術(shù)擁有Java語(yǔ)言“一次編寫(xiě),各處執(zhí)行”的特點(diǎn)。隨著越來(lái)越多的供貨商將JavaServer Pages 技術(shù)添加到他們的產(chǎn)品中,您可以針對(duì)自己公司的需求,做出審慎評(píng)估后,選擇符合公司成本及規(guī)模的服務(wù)器,假若未來(lái)的需求有所變更時(shí),更換服務(wù)器平臺(tái)并不影響之前所投下的成本、人力所開(kāi)發(fā)的應(yīng)用程序。
          ● 搭配可重復(fù)使用的組件
          JavaServer Pages技術(shù)可依賴于重復(fù)使用跨平臺(tái)的組件(如:JavaBean或Enterprise JavaBean組件)來(lái)執(zhí)行更復(fù)雜的運(yùn)算、數(shù)據(jù)處理。開(kāi)發(fā)人員能夠共享開(kāi)發(fā)完成的組件,或者能夠加強(qiáng)這些組件的功能,讓更多用戶或是客戶團(tuán)體使用。基于善加利用組件的方法,可以加快整體開(kāi)發(fā)過(guò)程,也大大降低公司的開(kāi)發(fā)成本和人力。
          ● 采用標(biāo)簽化頁(yè)面開(kāi)發(fā)
          Web 網(wǎng)頁(yè)開(kāi)發(fā)人員不一定都是熟悉Java 語(yǔ)言的程序員。因此,JSP 技術(shù)能夠?qū)⒃S多功能封裝起來(lái),成為一個(gè)自定義的標(biāo)簽,這些功能是完全根據(jù)XML 的標(biāo)準(zhǔn)來(lái)制訂的,即JSP 技術(shù)中的標(biāo)簽庫(kù)(Tag Library)。因此,Web 頁(yè)面開(kāi)發(fā)人員可以運(yùn)用自定義好的標(biāo)簽來(lái)達(dá)成工作需求,而無(wú)須再寫(xiě)復(fù)雜的Java 語(yǔ)法,讓W(xué)eb 頁(yè)面開(kāi)發(fā)人員亦能快速開(kāi)發(fā)出一動(dòng)態(tài)內(nèi)容網(wǎng)頁(yè)。
          今后,第三方開(kāi)發(fā)人員和其他人員可以為常用功能建立自己的標(biāo)簽庫(kù),讓Web 網(wǎng)頁(yè)開(kāi)發(fā)人員能夠使用熟悉的開(kāi)發(fā)工具,如同HTML 一樣的標(biāo)簽語(yǔ)法來(lái)執(zhí)行特定功能的工作。
          N-tier 企業(yè)應(yīng)用架構(gòu)的支持
          有鑒于網(wǎng)際網(wǎng)絡(luò)的發(fā)展,為因應(yīng)未來(lái)服務(wù)越來(lái)越繁雜的要求,且不再受地域的限制,因此,
          必須放棄以往Client-Server的Two-tier 架構(gòu),進(jìn)而轉(zhuǎn)向更具威力、彈性的分散性對(duì)象系統(tǒng)。由于JavaServer Page 技術(shù)是Java 2 Platform Enterprise Edition (J2EE)集成中的一部分,它主要是負(fù)責(zé)前端顯示經(jīng)過(guò)復(fù)雜運(yùn)算后之結(jié)果內(nèi)容,而分散性的對(duì)象系統(tǒng)則是主要依賴EJB ( Enterprise JavaBean )和JNDI ( Java Naming and Directory Interface )構(gòu)建而成。

          (4)、JavaBean(Application)
          應(yīng)用組件技術(shù)
          Application
          是Java應(yīng)用程序,在WEB項(xiàng)目和一些開(kāi)發(fā)中主要應(yīng)用JavaBean。它就是Application的一部分,邏輯運(yùn)算能力很強(qiáng),能極大的發(fā)揮Java語(yǔ)言的優(yōu)點(diǎn)。JavaBean 被稱(chēng)為是Java 組件技術(shù)的核心。JavaBean 的結(jié)構(gòu)必須滿足一定的命名約定。JavaBean能提供常用功能并且可以重復(fù)使用,這使得開(kāi)發(fā)人員可以把某些關(guān)鍵功能和核心算法提取出來(lái)封裝成為一個(gè)組件對(duì)象,這樣就增加了代碼的重用率和系統(tǒng)的安全性。

          高級(jí)的WEB項(xiàng)目會(huì)應(yīng)用到以上所有技術(shù),它們之間聯(lián)合使用和協(xié)作開(kāi)發(fā)會(huì)提高開(kāi)發(fā)的效率和系統(tǒng)的性能。 

          2、面向?qū)ο蠓治鲈O(shè)計(jì)思想

          Java
          語(yǔ)言是完全面向?qū)ο蟮恼Z(yǔ)言,所以在項(xiàng)目設(shè)計(jì)時(shí)會(huì)有很大的幫助,在設(shè)計(jì)時(shí)應(yīng)盡量舍棄以往的面向過(guò)程的設(shè)計(jì)方式。

          在分析項(xiàng)目業(yè)務(wù)關(guān)系的時(shí)候,應(yīng)用一些UML(Unified Modeling Language)圖,例如常用的用例圖(use case diagram),類(lèi)圖(class diagram),時(shí)序圖(sequence diagram)等等,會(huì)有很大的幫助,這樣能盡快找出業(yè)務(wù)邏輯主要面對(duì)的對(duì)象,然后對(duì)每個(gè)對(duì)象進(jìn)行行為劃分,最后再實(shí)現(xiàn)對(duì)象之間的集成和通信。

          3、設(shè)計(jì)模式和框架結(jié)構(gòu)
          Java從語(yǔ)言角度來(lái)講不是很難,但是從整體設(shè)計(jì)角度來(lái)講我們還需要了解一些高級(jí)應(yīng)用框架。如果要設(shè)計(jì)一個(gè)良好的框架結(jié)構(gòu),單單只掌握J(rèn)ava語(yǔ)言遠(yuǎn)遠(yuǎn)不夠。這就涉及到一個(gè)設(shè)計(jì)模式,還有和設(shè)計(jì)模式相關(guān)的一些知識(shí)。

          設(shè)計(jì)模式在Java項(xiàng)目實(shí)施過(guò)程更是重中之重。主要在與兩層的設(shè)計(jì)模式、三層的設(shè)計(jì)模式和N層的設(shè)計(jì)模式。它直接決定著項(xiàng)目的應(yīng)用、部署和實(shí)際開(kāi)發(fā)設(shè)計(jì)。

          在普通的WEB項(xiàng)目中很多采用兩層的開(kāi)發(fā)結(jié)構(gòu)。JSP+Servlet或JSP+JavaBean。當(dāng)對(duì)開(kāi)發(fā)要求高的項(xiàng)目中使用很多的還是MVC的三層開(kāi)發(fā)結(jié)構(gòu),也就是JSP+Servlet+JavaBean。它能分有效的分離邏輯開(kāi)發(fā),使開(kāi)發(fā)人員能專(zhuān)注于各自的開(kāi)發(fā)。同時(shí)也能時(shí)整個(gè)開(kāi)發(fā)結(jié)構(gòu)流程更清晰,但是需要比較高的開(kāi)發(fā)配合度。
          在項(xiàng)目中,我們經(jīng)常使用著名的Model-View-Controller(MVC)架構(gòu)。
          MVC架構(gòu)是隨著smalltalk language語(yǔ)言的發(fā)展提出的,它是一個(gè)著名的用戶界面設(shè)計(jì)架構(gòu)。經(jīng)典的MVC架構(gòu)把一個(gè)組件(可認(rèn)為是整個(gè)應(yīng)用程序的一個(gè)模塊)劃分成三部分組 Model管理這個(gè)模塊中所用到的數(shù)據(jù)和業(yè)務(wù)邏輯。而View 管理模塊如何顯示給用戶,Controller 決定如何處理用戶和該模塊交互式時(shí)候產(chǎn)生的事件 如用戶點(diǎn)擊一個(gè)按鈕等。

          4、XML
          語(yǔ)言

          在服務(wù)器和設(shè)計(jì)模式結(jié)構(gòu)中會(huì)應(yīng)用到自定義文件,而且在應(yīng)用高級(jí)設(shè)計(jì)時(shí)也會(huì)定義自用的標(biāo)簽,現(xiàn)在流行的是用XML去定義配置,所以XML語(yǔ)言應(yīng)該有一定掌握。
          當(dāng)前,Java 2平臺(tái)企業(yè)版(J2EE)架構(gòu)在廠商市場(chǎng)和開(kāi)發(fā)者社區(qū)中倍受推崇。作為一種工具,可擴(kuò)展標(biāo)記語(yǔ)言(XML)簡(jiǎn)化了數(shù)據(jù)交換、進(jìn)程間消息交換這一類(lèi)的事情,因而對(duì)開(kāi)發(fā)者逐漸變得有吸引力,并開(kāi)始流行起來(lái)。自然,在J2EE架構(gòu)中訪問(wèn)或集成XML解決方案的想法也很誘人。因?yàn)檫@將是強(qiáng)大系統(tǒng)架構(gòu)同高度靈活的數(shù)據(jù)管理方案的結(jié)合。

          XML的應(yīng)用似乎是無(wú)窮無(wú)盡的,但它們大致上可以分為三大類(lèi):
          1、簡(jiǎn)單數(shù)據(jù)的表示和交換(針對(duì)XML的簡(jiǎn)單API(SAX)和文檔對(duì)象模型(DOM)語(yǔ)法解析,不同的文檔類(lèi)型定義(DTDs)和概要(schemas))
          2、用戶界面相關(guān)、表示相關(guān)的上下文(可擴(kuò)展樣式表語(yǔ)言(XSL),可擴(kuò)展樣式表語(yǔ)言轉(zhuǎn)換(XSLT))
          3、面向消息的計(jì)算(XML-RPC(遠(yuǎn)程過(guò)程調(diào)用),基于SOAP協(xié)議的Web 服務(wù)(Web Services),電子化業(yè)務(wù)XML(ebXML))

          5、網(wǎng)頁(yè)腳本語(yǔ)言
           
          為了提高WEB項(xiàng)目的整體性能,提高人機(jī)交互的友好界面,網(wǎng)頁(yè)的腳本語(yǔ)言是很有用處的,有的時(shí)候可以解決很大的難題或提高程序的性能和應(yīng)用性。

          網(wǎng)頁(yè)腳本語(yǔ)言的執(zhí)行都是在客戶端執(zhí)行的,速度很很快,并且大多的操作與服務(wù)器沒(méi)有交互運(yùn)算,所以在一些應(yīng)用中非常理想。在設(shè)計(jì)WEB項(xiàng)目的應(yīng)用中,網(wǎng)頁(yè)的腳本語(yǔ)言起著不能忽視的作用,所以如果設(shè)計(jì)WEB項(xiàng)目的應(yīng)用中,對(duì)JavaScript應(yīng)有一定的了解。
           
          JavaScript是一種基于對(duì)象(Object Based)和事件驅(qū)動(dòng)(Event Driven)并具有安全性能(Secure)的腳本語(yǔ)言。使用它的目的是與HTML超文本標(biāo)記語(yǔ)言、Java 腳本語(yǔ)言(Java小程序)一起實(shí)現(xiàn)在一個(gè)Web頁(yè)面中鏈接多個(gè)對(duì)象,與Web客戶交互作用。從而可以開(kāi)發(fā)客戶端的應(yīng)用程序等。它是通過(guò)嵌入或調(diào)入在標(biāo)準(zhǔn)的HTML語(yǔ)言中實(shí)現(xiàn)的。它具有以下幾個(gè)基本特點(diǎn):
          1.它是一種腳本編寫(xiě)語(yǔ)言
          JavaScript是一種腳本語(yǔ)言,它采用小程序段的方式實(shí)現(xiàn)編程。像其它腳本語(yǔ)言一樣,JavaScript同樣已是一種解釋性語(yǔ)言,它提供了一個(gè)易的開(kāi)發(fā)過(guò)程。
          它的基本結(jié)構(gòu)形式與C、C++、VB十分類(lèi)似。但它不像這些語(yǔ)言一樣,需要先編譯,而是在程序運(yùn)行過(guò)程中被逐行地解釋。它與HTML標(biāo)識(shí)結(jié)合在一起,從而方便用戶的使用操作。
          2. 基于對(duì)象的語(yǔ)言。
           JavaScript是一種基于對(duì)象的語(yǔ)言,同時(shí)以可以看作一種面向?qū)ο蟮摹_@意味著它能運(yùn)用自己已經(jīng)創(chuàng)建的對(duì)象。因此,許多功能可以來(lái)自于腳本環(huán)境中對(duì)象的方法與腳本的相互作用。
          3.簡(jiǎn)單性
           JavaScript的簡(jiǎn)單性主要體現(xiàn)在:首先它是一種基于Java基本語(yǔ)句和控制流之上的簡(jiǎn)單而緊湊的設(shè)計(jì), 從而對(duì)于學(xué)習(xí)Java是一種非常好的過(guò)渡。其次它的變量類(lèi)型是采用弱類(lèi)型,并未使用嚴(yán)格的數(shù)據(jù)類(lèi)型。
          4.安全性
           JavaScript是一種安全性語(yǔ)言,它不允許訪問(wèn)本地的硬盤(pán),并不能將數(shù)據(jù)存入到服務(wù)器上,不允許對(duì)網(wǎng)絡(luò)文檔進(jìn)行修改和刪除,只能通過(guò)瀏覽器實(shí)現(xiàn)信息瀏覽或動(dòng)態(tài)交互。從而有效地防止數(shù)據(jù)的丟失。
          5. 動(dòng)態(tài)性
           JavaScript是動(dòng)態(tài)的,它可以直接對(duì)用戶或客戶輸入做出響應(yīng),無(wú)須經(jīng)過(guò)Web服務(wù)程序。它對(duì)用戶的響應(yīng),是采用以事件驅(qū)動(dòng)的方式進(jìn)行的。所謂事件驅(qū)動(dòng),就是指在主頁(yè)(Home Page)中執(zhí)行了某種操作所產(chǎn)生的動(dòng)作,就稱(chēng)為“事件”(Event)。比如按下鼠標(biāo)、移動(dòng)窗口、選擇菜單等都可以視為事件。當(dāng)事件發(fā)生后,可能會(huì)引起相應(yīng)的事件響應(yīng)。

          6、開(kāi)發(fā)工具
          (1)、數(shù)據(jù)庫(kù)
          在主要的應(yīng)用中,數(shù)據(jù)庫(kù)相關(guān)的環(huán)節(jié)應(yīng)用很多,所以對(duì)數(shù)據(jù)庫(kù)應(yīng)該有一定了解。不能單單只了解一種數(shù)據(jù)庫(kù),因?yàn)樵诤芏鄬?shí)際開(kāi)發(fā)中會(huì)提出很多數(shù)據(jù)庫(kù)解決方案,所以只有在了解多種數(shù)據(jù)庫(kù)的情況下才能有一個(gè)比較方案。
          對(duì)于數(shù)據(jù)庫(kù)應(yīng)該了解他的性能和一些基本的操作常識(shí),還有該數(shù)據(jù)庫(kù)的特點(diǎn)。而針對(duì)與Java語(yǔ)言WEB項(xiàng)目的數(shù)據(jù)庫(kù)開(kāi)發(fā)則主要是對(duì)JDBC的應(yīng)用,還有數(shù)據(jù)庫(kù)事務(wù)處理和連接池等高級(jí)概念的應(yīng)用。

          (2)、Web服務(wù)器
           
          同數(shù)據(jù)庫(kù)一樣,應(yīng)該了解該服務(wù)器的性能,特點(diǎn)和一些常識(shí)。
          在應(yīng)用方面,Web服務(wù)器主要是針對(duì)于配置和部署,對(duì)目錄的配置,調(diào)試;對(duì)配置文件屬性的修改;對(duì)訪問(wèn)權(quán)限和并發(fā)性的控制;Java類(lèi)的部署等。

          (3)、集成開(kāi)發(fā)環(huán)境(IDE):
          公欲善其事, 必先利其器”. 對(duì)于Web應(yīng)用開(kāi)發(fā)人員來(lái)講,好的集成開(kāi)發(fā)環(huán)境(IDE:Integrated Development Enviroment)是非常重要的。目前在市場(chǎng)上占主導(dǎo)位置的一個(gè)集成開(kāi)發(fā)工具就是Eclipse.
           (轉(zhuǎn)載文章請(qǐng)保留出處:Java家(www.javajia.com))
          posted @ 2007-07-02 12:58 和田雨 閱讀(340) | 評(píng)論 (0)編輯 收藏

          一些網(wǎng)站采用了字母和數(shù)字的驗(yàn)證碼,數(shù)字和字母加起來(lái)一共30多個(gè),如果有心,還是能夠通過(guò)方法識(shí)別出來(lái)。 O8<@d3  
          我在網(wǎng)上看到一篇文章"jsp彩色驗(yàn)證碼",我進(jìn)行了加強(qiáng),生成的驗(yàn)證碼內(nèi)容為漢字,可以方便應(yīng)用在面向漢語(yǔ)網(wǎng)民的網(wǎng)站認(rèn)證上。
          oxcd:, Vc  
          iXV6:&\9"r  
          當(dāng)然,我還看到別人考慮的幾種方法:
          {i867@H_e  
          1.生成計(jì)算題,比如20+34*(23-12)=?,用戶必須輸入正確的答案才能通過(guò)驗(yàn)證,這個(gè)想法挺好,但是用戶必須打開(kāi)計(jì)算器進(jìn)行計(jì)算,增加了用戶的難度
          dza^8ynu  
          2.問(wèn)一些常見(jiàn)的問(wèn)題,比如“人”這個(gè)字由幾筆組成。這也是不錯(cuò)的想法,關(guān)鍵是必須有上萬(wàn)條的題庫(kù),而且題庫(kù)的答案必須是明確的,簡(jiǎn)單易比較的,這個(gè)難度也較大。
          |||o<  
          ejas+  
          下面是彩色漢字驗(yàn)證碼的代碼。
          ,"-07[X  
          ]j&J&.:&,  
          dVTACb`  
          <%@ page contentType="image/jpeg" import="java.awt.*,java.awt.image.*,java.util.*,javax.imageio.*" %>
          YwpC}G;   
          <%!
          (.&vEwK  
          //create by smallnest
          z(S Lcx  
          //email:
          smallnest@gmail.com <p-Y a_&u  
          //website:
          www.kuaff.com hpC%no /s  
          Du:x%[~]  
          99Ztr4E  
          Jk)|S2  
          : pZ<K"Z-&  
          //生成隨機(jī)顏色
          s*# ]mm.MD  
          Color getRandColor(Random random,int fc,int bc)
          M-C|5ahf8B  
          {
          !DF}` d  
          if(fc>255) fc=255;
          {+9C59]sA  
          if(bc>255) bc=255;
          ^k41]4 RD  
          int r=fc+random.nextInt(bc-fc);
          ')/u5Y{fp  
          int g=fc+random.nextInt(bc-fc);
          F\(TcR(  
          int b=fc+random.nextInt(bc-fc);
          qQ5> k=3  
          return new Color(r,g,b);
          2Es&*Kr+"  
          }
           '0 )!e!=  
          %>
          =H NQZD`e  
          <%
          q=X0lJ<V  
          //設(shè)置頁(yè)面不緩存
          Qsd{m_y#  
          response.setHeader("Pragma","No-cache");
          p[brf>x  
          response.setHeader("Cache-Control","no-cache");
          h-r<TW  
          response.setDateHeader("Expires", 0);
          hB~(UCtd  
          {m@moH 3N  
          ">^0|,(e  
          // 設(shè)置圖片的長(zhǎng)寬
          G"KO  
          int width=176, height=30;
          rl> 1bV?8  
          //設(shè)置備選漢字,剔除一些不雅的漢字
          >22XQgGh  
          String base = "\u7684\u4e00\u4e86\u662f\u6211\u4e0d\u5728\u4eba\u4eec\u6709\u6765\u4ed6\u8fd9\u4e0a\u7740\u4e2a\u5730\u5230\u5927\u91cc\u8bf4\u5c31\u53bb\u5b50\u5f97\u4e5f\u548c\u90a3\u8981\u4e0b\u770b\u5929\u65f6\u8fc7\u51fa\u5c0f\u4e48\u8d77\u4f60\u90fd\u628a\u597d\u8fd8\u591a\u6ca1\u4e3a\u53c8\u53ef\u5bb6\u5b66\u53ea\u4ee5\u4e3b\u4f1a\u6837\u5e74\u60f3\u751f\u540c\u8001\u4e2d\u5341\u4ece\u81ea\u9762\u524d\u5934\u9053\u5b83\u540e\u7136\u8d70\u5f88\u50cf\u89c1\u4e24\u7528\u5979\u56fd\u52a8\u8fdb\u6210\u56de\u4ec0\u8fb9\u4f5c\u5bf9\u5f00\u800c\u5df1\u4e9b\u73b0\u5c71\u6c11\u5019\u7ecf\u53d1\u5de5\u5411\u4e8b\u547d\u7ed9\u957f\u6c34\u51e0\u4e49\u4e09\u58f0\u4e8e\u9ad8\u624b\u77e5\u7406\u773c\u5fd7\u70b9\u5fc3\u6218\u4e8c\u95ee\u4f46\u8eab\u65b9\u5b9e\u5403\u505a\u53eb\u5f53\u4f4f\u542c\u9769\u6253\u5462\u771f\u5168\u624d\u56db\u5df2\u6240\u654c\u4e4b\u6700\u5149\u4ea7\u60c5\u8def\u5206\u603b\u6761\u767d\u8bdd\u4e1c\u5e2d\u6b21\u4eb2\u5982\u88ab\u82b1\u53e3\u653e\u513f\u5e38\u6c14\u4e94\u7b2c\u4f7f\u5199\u519b\u5427\u6587\u8fd0\u518d\u679c\u600e\u5b9a\u8bb8\u5feb\u660e\u884c\u56e0\u522b\u98de\u5916\u6811\u7269\u6d3b\u90e8\u95e8\u65e0\u5f80\u8239\u671b\u65b0\u5e26\u961f\u5148\u529b\u5b8c\u5374\u7ad9\u4ee3\u5458\u673a\u66f4\u4e5d\u60a8\u6bcf\u98ce\u7ea7\u8ddf\u7b11\u554a\u5b69\u4e07\u5c11\u76f4\u610f\u591c\u6bd4\u9636\u8fde\u8f66\u91cd\u4fbf\u6597\u9a6c\u54ea\u5316\u592a\u6307\u53d8\u793e\u4f3c\u58eb\u8005\u5e72\u77f3\u6ee1\u65e5\u51b3\u767e\u539f\u62ff\u7fa4\u7a76\u5404\u516d\u672c\u601d\u89e3\u7acb\u6cb3\u6751\u516b\u96be\u65e9\u8bba\u5417\u6839\u5171\u8ba9\u76f8\u7814\u4eca\u5176\u4e66\u5750\u63a5\u5e94\u5173\u4fe1\u89c9\u6b65\u53cd\u5904\u8bb0\u5c06\u5343\u627e\u4e89\u9886\u6216\u5e08\u7ed3\u5757\u8dd1\u8c01\u8349\u8d8a\u5b57\u52a0\u811a\u7d27\u7231\u7b49\u4e60\u9635\u6015\u6708\u9752\u534a\u706b\u6cd5\u9898\u5efa\u8d76\u4f4d\u5531\u6d77\u4e03\u5973\u4efb\u4ef6\u611f\u51c6\u5f20\u56e2\u5c4b\u79bb\u8272\u8138\u7247\u79d1\u5012\u775b\u5229\u4e16\u521a\u4e14\u7531\u9001\u5207\u661f\u5bfc\u665a\u8868\u591f\u6574\u8ba4\u54cd\u96ea\u6d41\u672a\u573a\u8be5\u5e76\u5e95\u6df1\u523b\u5e73\u4f1f\u5fd9\u63d0\u786e\u8fd1\u4eae\u8f7b\u8bb2\u519c\u53e4\u9ed1\u544a\u754c\u62c9\u540d\u5440\u571f\u6e05\u9633\u7167\u529e\u53f2\u6539\u5386\u8f6c\u753b\u9020\u5634\u6b64\u6cbb\u5317\u5fc5\u670d\u96e8\u7a7f\u5185\u8bc6\u9a8c\u4f20\u4e1a\u83dc\u722c\u7761\u5174\u5f62\u91cf\u54b1\u89c2\u82e6\u4f53\u4f17\u901a\u51b2\u5408\u7834\u53cb\u5ea6\u672f\u996d\u516c\u65c1\u623f\u6781\u5357\u67aa\u8bfb\u6c99\u5c81\u7ebf\u91ce\u575a\u7a7a\u6536\u7b97\u81f3\u653f\u57ce\u52b3\u843d\u94b1\u7279\u56f4\u5f1f\u80dc\u6559\u70ed\u5c55\u5305\u6b4c\u7c7b\u6e10\u5f3a\u6570\u4e61\u547c\u6027\u97f3\u7b54\u54e5\u9645\u65e7\u795e\u5ea7\u7ae0\u5e2e\u5566\u53d7\u7cfb\u4ee4\u8df3\u975e\u4f55\u725b\u53d6\u5165\u5cb8\u6562\u6389\u5ffd\u79cd\u88c5\u9876\u6025\u6797\u505c\u606f\u53e5\u533a\u8863\u822c\u62a5\u53f6\u538b\u6162\u53d4\u80cc\u7ec6";
          =-/!18  
          //備選漢字的長(zhǎng)度
          k63GRbd?  
          int length = base.length();
          n\+47N*ze  
          ,"jx!  
          2b8~3d[L  
          //創(chuàng)建內(nèi)存圖像
          e/6VUo@>%  
          BufferedImage image = new BufferedImage(width, height, BufferedImage.TYPE_INT_RGB);
          X/$V%6)  
          // 獲取圖形上下文
          "g&Xk0  
          Graphics g = image.getGraphics();
          NEE4{?Q?  
          k >@l  
          Q{AR6CRAt  
          //創(chuàng)建隨機(jī)類(lèi)的實(shí)例
          ch6) "2  
          Random random = new Random();
          >>#>"<CMT  
          K#]~p^Z}  
          aF@:;!cVu  
          // 設(shè)定圖像背景色(因?yàn)槭亲霰尘埃云?
          NAL2w3  
          g.setColor(getRandColor(random,200,250));
          {<=!bYh  
          g.fillRect(0, 0, width, height);
          7@mFmb] -#  
          +.S``"'  
          Eg^${%,d  
          //備選字體
          RE]p3t|O  
          String[] fontTypes = {"\u5b8b\u4f53","\u65b0\u5b8b\u4f53","\u9ed1\u4f53","\u6977\u4f53","\u96b6\u4e66"};
          b<psU?/% _  
          int fontTypesLength = fontTypes.length;
          YZ?KfO=r  
          6< 1B;  
          <;G d~(  
          //在圖片背景上增加噪點(diǎn)
          Q`6 A{@5Y  
          g.setColor(getRandColor(random,160,200));
          8M`w  
          g.setFont(new Font("Times New Roman",Font.PLAIN,14));
          S h (%oy<+  
          for (int i=0;i<6;i++)
          [NFmqEEanb  
          {
          _R6 Lr9q  
          g.drawString("*********************************************",0,5*(i+2));
          goQ2Va+  
          }
          }):f;M*  
          9L+)0)(  
          +4vsFa6*  
          Fj%N4$?(m  
          +Sv2qb  
          //取隨機(jī)產(chǎn)生的認(rèn)證碼(6個(gè)漢字)
          %[B|3.e  
          =e?R(E^Y"  
          XSp~5|I  
          //保存生成的漢字字符串
          p^ >Q(g  
          String sRand="";
          /0A5(P&q/  
          for (int i=0;i<6;i++)
          ypeul27H8  
          {
          h570L^  
          int start = random.nextInt(length);
          L[4Aolk:M  
          String rand=base.substring(start,start+1);
          &hA0:^  
          sRand+=rand;
          D/ ^Kl  
          $N^odf(2   
          //設(shè)置字體的顏色
          2A<BW~3FPt  
          g.setColor(getRandColor(random,10,150));
          C hhWE@!  
          //設(shè)置字體
          6<\w'-  
          g.setFont(new Font(fontTypes[random.nextInt(fontTypesLength)],Font.BOLD,18 + random.nextInt(6)));
          ftVk!3e  
          //將此漢字畫(huà)到圖片上
          dMjL#H}  
          g.drawString(rand,24*i+ 10 + random.nextInt(8),24);
          I_i 7 %  
          }
          ! [ $qI1  
          X8H6Fki  
          eP{42 C0=  
          //將認(rèn)證碼存入session
          kCl -?N{og  
          session.setAttribute("rand",sRand);
          <uT}CUy  
          lfKXsvP  
          +M'E6E%w  
          g.dispose();
          *$4SR?  
          OIFYpaBXb  
          <~[q5tC+  
          //輸出圖象到頁(yè)面
          /z <4A^+  
          ImageIO.write(image, "JPEG", response.getOutputStream());
          L$ "5TP<  
          %>
          &+ Q :0Y  
          ~ oNku2,  
           |D6AG!J  
          4f$X#:  
          ===========================
          qozOA`,l  
          /j|Au)P  
          注:此方法沒(méi)有在本站JSP虛擬主機(jī)測(cè)試過(guò),不知道性能如何,如果要用驗(yàn)證請(qǐng)看:
          ~J|X4>.Vm  
          posted @ 2007-07-02 12:57 和田雨 閱讀(694) | 評(píng)論 (0)編輯 收藏

          僅列出標(biāo)題
          共9頁(yè): 上一頁(yè) 1 2 3 4 5 6 7 8 9 下一頁(yè) 
          主站蜘蛛池模板: 英德市| 新安县| 铁岭县| 山东省| 翼城县| 衡水市| 偏关县| 苗栗市| 大邑县| 阜宁县| 清水河县| 磴口县| 布尔津县| 玛曲县| 南昌市| 温泉县| 泰顺县| 沅陵县| 阿图什市| 南昌县| 阳山县| 美姑县| 达州市| 阆中市| 班戈县| 苍溪县| 冕宁县| 丰城市| 罗源县| 耒阳市| 紫阳县| 平南县| 临清市| 蒙阴县| 乳山市| 天镇县| 岳池县| 沈阳市| 筠连县| 庆城县| 柏乡县|