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

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

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

    隨筆 - 9, 文章 - 0, 評論 - 3, 引用 - 0
    數(shù)據(jù)加載中……

    2007年11月7日

    初探spring applicationContext在web容器中加載過程

         摘要: 轉(zhuǎn)載自www.javaworld.com.tw 袁杰 原文 首先從WEB.XML入手 ==>web.xml 1 2 3 4 5 6 ...  閱讀全文

    posted @ 2007-11-07 15:06 空杯 閱讀(10739) | 評論 (1)編輯 收藏

    2007年10月29日

    JUnit 4 搶先看

    from: http://www.ibm.com/developerworks/cn/java/j-junit4.html

    JUnit 由 Kent Beck 和 Erich Gamma 開發(fā),幾乎毫無疑問是迄今所開發(fā)的最重要的第三方 Java 庫。正如 Martin Fowler 所說,“在軟件開發(fā)領(lǐng)域,從來就沒有如此少的代碼起到了如此重要的作用”。JUnit 引導(dǎo)并促進了測試的盛行。由于 JUnit,Java 代碼變得更健壯,更可靠,bug 也比以前更少。JUnit(它本身的靈感來自 Smalltalk 的 SUnit)衍生了許多 xUnit 工具,將單元測試的優(yōu)勢應(yīng)用于各種語言。nUnit (.NET)、pyUnit (Python)、CppUnit (C++)、dUnit (Delphi) 以及其他工具,影響了各種平臺和語言上的程序員的測試工作。

    然而,JUnit 僅僅是一個工具而已。真正的優(yōu)勢來自于 JUnit 所采用的思想和技術(shù),而不是框架本身。單元測試、測試先行的編程和測試驅(qū)動的開發(fā)并非都要在 JUnit 中實現(xiàn),任何比較 GUI 的編程都必須用 Swing 來完成。JUnit 本身的最后一次更新差不多是三年以前了。盡管它被證明比大多數(shù)框架更健壯、更持久,但是也發(fā)現(xiàn)了 bug;而更重要的是,Java 不斷在發(fā)展。Java 語言現(xiàn)在支持泛型、枚舉、可變長度參數(shù)列表和注釋,這些特性為可重用的框架設(shè)計帶來了新的可能。

    JUnit 的停滯不前并沒有被那些想要廢棄它的程序員所打敗。挑戰(zhàn)者包括 Bill Venners 的 Artima SuiteRunner 以及 Cedric Beust 的 TestNG 等。這些庫有一些可圈可點的特性,但是都沒有達到 JUnit 的知名度和市場占有份額。它們都沒有在諸如 Ant、Maven 或 Eclipse 之類的產(chǎn)品中具有廣泛的開箱即用支持。所以 Beck 和 Gamma 著手開發(fā)了一個新版本的 JUnit,它利用 Java 5 的新特性(尤其是注釋)的優(yōu)勢,使得單元測試比起用最初的 JUnit 來說更加簡單。用 Beck 的話來說,“JUnit 4 的主題是通過進一步簡化 JUnit,鼓勵更多的開發(fā)人員編寫更多的測試。”JUnit 4 盡管保持了與現(xiàn)有 JUnit 3.8 測試套件的向后兼容,但是它仍然承諾是自 JUnit 1.0 以來 Java 單元測試方面最重大的改進。

    注意:該框架的改進是相當前沿的。盡管 JUnit 4 的大輪廓很清晰,但是其細節(jié)仍然可以改變。這意味著本文是對 JUnit 4 搶先看,而不是它的最終效果。

    測試方法

    以前所有版本的 JUnit 都使用命名約定和反射來定位測試。例如,下面的代碼測試 1+1 等于 2:

    import junit.framework.TestCase;
                public class AdditionTest extends TestCase {
                private int x = 1;
                private int y = 1;
                public void testAddition() {
                int z = x + y;
                assertEquals(2, z);
                }
                }

    而在 JUnit 4 中,測試是由 @Test 注釋來識別的,如下所示:

    import org.junit.Test;
                import junit.framework.TestCase;
                public class AdditionTest extends TestCase {
                private int x = 1;
                private int y = 1;
                @Test public void testAddition() {
                int z = x + y;
                assertEquals(2, z);
                }
                }

    使用注釋的優(yōu)點是不再需要將所有的方法命名為 testFoo()testBar(),等等。例如,下面的方法也可以工作:

    import org.junit.Test;
                import junit.framework.TestCase;
                public class AdditionTest extends TestCase {
                private int x = 1;
                private int y = 1;
                @Test public void additionTest() {
                int z = x + y;
                assertEquals(2, z);
                }
                }

    下面這個方法也同樣能夠工作:

    import org.junit.Test;
                import junit.framework.TestCase;
                public class AdditionTest extends TestCase {
                private int x = 1;
                private int y = 1;
                @Test public void addition() {
                int z = x + y;
                assertEquals(2, z);
                }
                }

    這允許您遵循最適合您的應(yīng)用程序的命名約定。例如,我介紹的一些例子采用的約定是,測試類對其測試方法使用與被測試的類相同的名稱。例如,List.contains()ListTest.contains() 測試,List.add()ListTest.addAll() 測試,等等。

    TestCase 類仍然可以工作,但是您不再需要擴展它了。只要您用 @Test 來注釋測試方法,就可以將測試方法放到任何類中。但是您需要導(dǎo)入 junit.Assert 類以訪問各種 assert 方法,如下所示:

    import org.junit.Assert;
                public class AdditionTest {
                private int x = 1;
                private int y = 1;
                @Test public void addition() {
                int z = x + y;
                Assert.assertEquals(2, z);
                }
                }

    您也可以使用 JDK 5 中新特性(static import),使得與以前版本一樣簡單:

    import static org.junit.Assert.assertEquals;
                public class AdditionTest {
                private int x = 1;
                private int y = 1;
                @Test public void addition() {
                int z = x + y;
                assertEquals(2, z);
                }
                }

    這種方法使得測試受保護的方法非常容易,因為測試案例類現(xiàn)在可以擴展包含受保護方法的類了。





    回頁首


    SetUp 和 TearDown

    JUnit 3 測試運行程序(test runner)會在運行每個測試之前自動調(diào)用 setUp() 方法。該方法一般會初始化字段,打開日志記錄,重置環(huán)境變量,等等。例如,下面是摘自 XOM 的 XSLTransformTest 中的 setUp() 方法:

    protected void setUp() {
                System.setErr(new PrintStream(new ByteArrayOutputStream()));
                inputDir = new File("data");
                inputDir = new File(inputDir, "xslt");
                inputDir = new File(inputDir, "input");
                }

    在 JUnit 4 中,您仍然可以在每個測試方法運行之前初始化字段和配置環(huán)境。然而,完成這些操作的方法不再需要叫做 setUp(),只要用 @Before 注釋來指示即可,如下所示:

    @Before protected void initialize() {
                System.setErr(new PrintStream(new ByteArrayOutputStream()));
                inputDir = new File("data");
                inputDir = new File(inputDir, "xslt");
                inputDir = new File(inputDir, "input");
                }

    甚至可以用 @Before 來注釋多個方法,這些方法都在每個測試之前運行:

    @Before protected void findTestDataDirectory() {
                inputDir = new File("data");
                inputDir = new File(inputDir, "xslt");
                inputDir = new File(inputDir, "input");
                }
                @Before protected void redirectStderr() {
                System.setErr(new PrintStream(new ByteArrayOutputStream()));
                }

    清除方法與此類似。在 JUnit 3 中,您使用 tearDown() 方法,該方法類似于我在 XOM 中為消耗大量內(nèi)存的測試所使用的方法:

    protected void tearDown() {
                doc = null;
                System.gc();
                } 

    對于 JUnit 4,我可以給它取一個更自然的名稱,并用 @After 注釋它:

    @After protected void disposeDocument() {
                doc = null;
                System.gc();
                } 

    @Before 一樣,也可以用 @After 來注釋多個清除方法,這些方法都在每個測試之后運行。

    最后,您不再需要在超類中顯式調(diào)用初始化和清除方法,只要它們不被覆蓋即可,測試運行程序?qū)⒏鶕?jù)需要自動為您調(diào)用這些方法。超類中的 @Before 方法在子類中的 @Before 方法之前被調(diào)用(這反映了構(gòu)造函數(shù)調(diào)用的順序)。@After 方法以反方向運行:子類中的方法在超類中的方法之前被調(diào)用。否則,多個 @Before@After 方法的相對順序就得不到保證。

    套件范圍的初始化

    JUnit 4 也引入了一個 JUnit 3 中沒有的新特性:類范圍的 setUp()tearDown() 方法。任何用 @BeforeClass 注釋的方法都將在該類中的測試方法運行之前剛好運行一次,而任何用 @AfterClass 注釋的方法都將在該類中的所有測試都運行之后剛好運行一次。

    例如,假設(shè)類中的每個測試都使用一個數(shù)據(jù)庫連接、一個網(wǎng)絡(luò)連接、一個非常大的數(shù)據(jù)結(jié)構(gòu),或者還有一些對于初始化和事情安排來說比較昂貴的其他資源。不要在每個測試之前都重新創(chuàng)建它,您可以創(chuàng)建它一次,并還原它一次。該方法將使得有些測試案例運行起來快得多。例如,當我測試調(diào)用第三方庫的代碼中的錯誤處理時,我通常喜歡在測試開始之前重定向 System.err,以便輸出不被預(yù)期的錯誤消息打亂。然后我在測試結(jié)束后還原它,如下所示:

    // This class tests a lot of error conditions, which
                // Xalan annoyingly logs to System.err. This hides System.err
                // before each test and restores it after each test.
                private PrintStream systemErr;
                @BeforeClass protected void redirectStderr() {
                systemErr = System.err; // Hold on to the original value
                System.setErr(new PrintStream(new ByteArrayOutputStream()));
                }
                @AfterClass protected void tearDown() {
                // restore the original value
                System.setErr(systemErr);
                }

    沒有必要在每個測試之前和之后都這樣做。但是一定要小心對待這個特性。它有可能會違反測試的獨立性,并引入非預(yù)期的混亂。如果一個測試在某種程度上改變了 @BeforeClass 所初始化的一個對象,那么它有可能會影響其他測試的結(jié)果。它有可能在測試套件中引入順序依賴,并隱藏 bug。與任何優(yōu)化一樣,只在剖析和基準測試證明您具有實際的問題之后才實現(xiàn)這一點。這就是說,我看到了不止一個測試套件運行時間如此之長,以至不能像它所需要的那樣經(jīng)常運行,尤其是那些需要建立很多網(wǎng)絡(luò)和數(shù)據(jù)庫連接的測試。(例如,LimeWire 測試套件運行時間超過兩小時。)要加快這些測試套件,以便程序員可以更加經(jīng)常地運行它們,您可以做的就是減少 bug。





    回頁首


    測試異常

    異常測試是 JUnit 4 中的最大改進。舊式的異常測試是在拋出異常的代碼中放入 try 塊,然后在 try 塊的末尾加入一個 fail() 語句。例如,該方法測試被零除拋出一個 ArithmeticException

    public void testDivisionByZero() {
                try {
                int n = 2 / 0;
                fail("Divided by zero!");
                }
                catch (ArithmeticException success) {
                assertNotNull(success.getMessage());
                }
                }

    該方法不僅難看,而且試圖挑戰(zhàn)代碼覆蓋工具,因為不管測試是通過還是失敗,總有一些代碼不被執(zhí)行。在 JUnit 4 中,您現(xiàn)在可以編寫拋出異常的代碼,并使用注釋來聲明該異常是預(yù)期的:

    @Test(expected=ArithmeticException.class)
                public void divideByZero() {
                int n = 2 / 0;
                }

    如果該異常沒有拋出(或者拋出了一個不同的異常),那么測試就將失敗。但是如果您想要測試異常的詳細消息或其他屬性,則仍然需要使用舊式的 try-catch 樣式。





    回頁首


    被忽略的測試

    也許您有一個測試運行的時間非常地長。不是說這個測試應(yīng)該運行得更快,而是說它所做的工作從根本上比較復(fù)雜或緩慢。需要訪問遠程網(wǎng)絡(luò)服務(wù)器的測試通常都屬于這一類。如果您不在做可能會中斷該類測試的事情,那么您可能想要跳過運行時間長的測試方法,以縮短編譯-測試-調(diào)試周期。或者也許是一個因為超出您的控制范圍的原因而失敗的測試。例如,W3C XInclude 測試套件測試 Java 還不支持的一些 Unicode 編碼的自動識別。不必老是被迫盯住那些紅色波浪線,這類測試可以被注釋為 @Ignore,如下所示:

    // Java doesn't yet support
                // the UTF-32BE and UTF32LE encodings
                @Ignore public void testUTF32BE()
                throws ParsingException, IOException, XIncludeException {
                File input = new File(
                "data/xinclude/input/UTF32BE.xml"
                );
                Document doc = builder.build(input);
                Document result = XIncluder.resolve(doc);
                Document expectedResult = builder.build(
                new File(outputDir, "UTF32BE.xml")
                );
                assertEquals(expectedResult, result);
                }

    測試運行程序?qū)⒉贿\行這些測試,但是它會指出這些測試被跳過了。例如,當使用文本界面時,會輸出一個“I”(代表 ignore),而不是為通過的測試輸出所經(jīng)歷的時間,也不是為失敗的測試輸出“E”:

    $ java -classpath .:junit.jar org.junit.runner.JUnitCore
                nu.xom.tests.XIncludeTest
                JUnit version 4.0rc1
                .....I..
                Time: 1.149
                OK (7 tests)

    但是一定要小心。最初編寫這些測試可能有一定的原因。如果永遠忽略這些測試,那么它們期望測試的代碼可能會中斷,并且這樣的中斷可能不能被檢測到。忽略測試只是一個權(quán)宜之計,不是任何問題的真正解決方案。





    回頁首


    時間測試

    測試性能是單元測試最為痛苦的方面之一。JUnit 4 沒有完全解決這個問題,但是它對這個問題有所幫助。測試可以用一個超時參數(shù)來注釋。如果測試運行的時間超過指定的毫秒數(shù),則測試失敗。例如,如果測試花費超過半秒時間去查找以前設(shè)置的一個文檔中的所有元素,那么該測試失敗:

    @Test(timeout=500) public void retrieveAllElementsInDocument() {
                doc.query("http://*");
                } 

    除了簡單的基準測試之外,時間測試也對網(wǎng)絡(luò)測試很有用。在一個測試試圖連接到的遠程主機或數(shù)據(jù)庫宕機或變慢時,您可以忽略該測試,以便不阻塞所有其他的測試。好的測試套件執(zhí)行得足夠快,以至程序員可以在每個測試發(fā)生重大變化之后運行這些測試,有可能一天運行幾十次。設(shè)置一個超時使得這一點更加可行。例如,如果解析 http://www.ibiblio.org/xml 花費了超過 2 秒,那么下面的測試就會超時:

    @Test(timeout=2000)
                public void remoteBaseRelativeResolutionWithDirectory()
                throws IOException, ParsingException {
                builder.build("http://www.ibiblio.org/xml");
                } 





    回頁首


    新的斷言

    JUnit 4 為比較數(shù)組添加了兩個 assert() 方法:

    public static void assertEquals(Object[] expected, Object[] actual)
                public static void assertEquals(String message, Object[] expected,
                Object[] actual)
                

    這兩個方法以最直接的方式比較數(shù)組:如果數(shù)組長度相同,且每個對應(yīng)的元素相同,則兩個數(shù)組相等,否則不相等。數(shù)組為空的情況也作了考慮。





    回頁首


    需要補充的地方

    JUnit 4 基本上是一個新框架,而不是舊框架的升級版本。JUnit 3 開發(fā)人員可能會找到一些原來沒有的特性。

    最明顯的刪節(jié)就是 GUI 測試運行程序。如果您想在測試通過時看到賞心悅目的綠色波浪線,或者在測試失敗時看到令人焦慮的紅色波浪線,那么您需要一個具有集成 JUnit 支持的 IDE,比如 Eclipse。不管是 Swing 還是 AWT 測試運行程序都不會被升級或捆綁到 JUnit 4 中。

    下一個驚喜是,失敗(assert 方法檢測到的預(yù)期的錯誤)與錯誤(異常指出的非預(yù)期的錯誤)之間不再有任何差別。盡管 JUnit 3 測試運行程序仍然可以區(qū)別這些情況,而 JUnit 4 運行程序?qū)⒉辉倌軌騾^(qū)分。

    最后,JUnit 4 沒有 suite() 方法,這些方法用于從多個測試類構(gòu)建一個測試套件。相反,可變長參數(shù)列表用于允許將不確定數(shù)量的測試傳遞給測試運行程序。

    我對消除了 GUI 測試運行程序并不感到太高興,但是其他更改似乎有可能增加 JUnit 的簡單性。只要考慮有多少文檔和 FAQ 當前專門用于解釋這幾點,然后考慮對于 JUnit 4,您不再需要解釋這幾點了。





    回頁首


    編譯和運行 JUnit 4

    當前,還沒有 JUnit 4 的庫版本。如果您想要體驗新的版本,那么您需要從 SourceForge 上的 CVS 知識庫獲取它。分支(branch)是“Version4”(參見 參考資料)。注意,很多的文檔沒有升級,仍然是指以舊式的 3.x 方式做事。Java 5 對于編譯 JUnit 4 是必需的,因為 JUnit 4 大量用到注釋、泛型以及 Java 5 語言級的其他特性。

    自 JUnit 3 以來,從命令行運行測試的語法發(fā)生了一點變化。您現(xiàn)在使用 org.junit.runner.JUnitCore 類:

    $ java -classpath .:junit.jar org.junit.runner.JUnitCore
                TestA TestB TestC...
                JUnit version 4.0rc1
                Time: 0.003
                OK (0 tests)

    兼容性

    Beck 和 Gamma 努力維持向前和向后兼容。JUnit 4 測試運行程序可以運行 JUnit 3 測試,不用做任何更改。只要將您想要運行的每個測試的全限定類名傳遞給測試運行程序,就像針對 JUnit 4 測試一樣。運行程序足夠智能,可以分辨出哪個測試類依賴于哪個版本的 JUnit,并適當?shù)卣{(diào)用它。

    向后兼容要困難一些,但是也可以在 JUnit 3 測試運行程序中運行 JUnit 4 測試。這一點很重要,所以諸如 Eclipse 之類具有集成 JUnit 支持的工具可以處理 JUnit 4,而不需要更新。為了使 JUnit 4 測試可以運行在 JUnit 3 環(huán)境中,可以將它們包裝在 JUnit4TestAdapter 中。將下面的方法添加到您的 JUnit 4 測試類中應(yīng)該就足夠了:

    public static junit.framework.Test suite() {
                return new JUnit4TestAdapter(AssertionTest.class);
                }

    但是由于 Java 比較多變,所以 JUnit 4 一點都不向后兼容。JUnit 4 完全依賴于 Java 5 特性。對于 Java 1.4 或更早版本,它將不會編譯或運行。





    回頁首


    前景

    JUnit 4 遠沒有結(jié)束。很多重要的方面沒有提及,包括大部分的文檔。我不推薦現(xiàn)在就將您的測試套件轉(zhuǎn)換成注釋和 JUnit 4。即使如此,開發(fā)仍在快速進行,并且 JUnit 4 前景非常看好。盡管 Java 2 程序員在可預(yù)見的未來仍然需要使用 JUnit 3.8,但是那些已經(jīng)轉(zhuǎn)移到 Java 5 的程序員則應(yīng)該很快考慮使他們的測試套件適合于這個新的框架,以便匹配。

    posted @ 2007-10-29 16:09 空杯 閱讀(303) | 評論 (0)編輯 收藏

    Spring 與Hibernate的延遲加載和Dao模式

         摘要: Hibernate 與延遲加載: Hibernate 對象關(guān)系映射提供延遲的與非延遲的對象初始化。非延遲加載在讀取一個對象的時候會將與這個對象所有相關(guān)的其他對象一起讀取出來。這有時會導(dǎo)致成百的(如果不是成千的話) select 語句在讀取對象的時候執(zhí)行。這個問題有時出現(xiàn)在使用雙向關(guān)系的時候,經(jīng)常會導(dǎo)致整個數(shù)據(jù)庫都在初始化的階段被讀出來了。當然,你可以不厭其煩地檢查每一個對象與其他對象的關(guān)系...  閱讀全文

    posted @ 2007-10-29 16:07 空杯 閱讀(814) | 評論 (1)編輯 收藏

    Hibernate的檢索方式

     

    Hibernate的檢索方式Hibernate的檢索方式:
     1.導(dǎo)航對象圖檢索方式
        根據(jù)已經(jīng)加載的對象,導(dǎo)航到其他對象.
     2.OID檢索方式
        按照對象的OID來檢索對象
     3.HQL檢索方式
        使用面向?qū)ο蟮腍QL查詢語言.
     4.QBC檢索方式
        使用QBC API來檢索對象,這種API封裝了基于字符串形式的查詢語句,提供了更加面向?qū)ο蟮慕涌?
        它主要由Criteria接口,Criterion接口和Expression類組合,它支持在運行時動態(tài)生成的查詢語句:
        以下程序代碼用于檢索姓名以字符"T"開頭,并且年齡為21的Customer對象:
          ///調(diào)用Session的createCriteria()方法創(chuàng)建一個Criteria對象
          Creteria criteria=session.createCriteria(Customer.class);
          //設(shè)置查詢條件,Expression類提供了一系列用于設(shè)定查詢條件的靜態(tài)方法,這些靜態(tài)方法都
           返回Criterion實例,每個Criterion實例代表一個查詢條件
          Criterion criterion1=Expression.like("name","T%");
          Criterion criterion2=Expression.eq("age",new Integer(21));
          ////Criteria的add()方法用于加入查詢條件.
          criteria=criteria.add(criterion1);
          criteria=criteria.add(criterion2);
          /////調(diào)用Criteria的list()方法執(zhí)行查詢語句,該方法返回List類型的查詢結(jié)果,在List集合中存放
            了符合查詢條件的持久化對象
          List result=criteria.list();
          對于以上代碼執(zhí)行的SQL語句為:select * from CUSTOMERS where NAME like"T%" and AGE=21;
          Criteria接口支持方法鏈編程風格,它的add()方法返回自身實例,而不是返回void類型
     5.本地SQL檢索方式:
     6.QBE檢索方式:
       它是QBC的子功能,QBE允許先創(chuàng)建一個對象樣板,然后檢索出所有和這個樣板相同的對象.如下:
        //創(chuàng)建一個CUstomer樣板對象
          Customer exampleCustomer=new Customer();
          exampleCustoemr.setAge(21);
          List result=session.createCriteria(Custoemr.class).add(Example.create(exampleCustomer)).list();
        因為QBE只支持"="和"like"比較運算符,所以一般采用HQL檢索方式或者QBC檢索方式.

    分頁查詢:
      Query和Criteria接口都提供了用于分頁顯示查詢結(jié)果的方法:
       setFirstResult(int firstResult):設(shè)定從哪一個對象開始檢索,參數(shù)firstResult表示這個對象在查詢結(jié)果中的索引位置,索引位置的起始值為0,
       在默認情況下,Query和Criteria接口從查詢結(jié)果中的第一個對象,也就是索引位置為0的對象開始檢索.
      setMaxResult(int maxResults):設(shè)定一次最多檢索出的對象數(shù)目,在默認情況下,Query和Criteria接口檢索出查詢接口中所有的對象

    檢查單個對象:
     Query和Criteria接口都提供了以下用于執(zhí)行查詢語句并返回查詢結(jié)果的方法:
    list()方法:返回一個List類型的查詢結(jié)果,在List集合中存放了所有滿足查詢條件的持久化對象
     uniqueResult()方法:返回單個對象.
     在某些情況下,如果只希望檢索出一個對象,可以先調(diào)用Query或Criteria接口的setMaxResult(1)方法,把最大檢索數(shù)目設(shè)為1,然后調(diào)用uniqueResult()方法,
     該方法返回一個Object類型的對象.
     // 采用HQL檢索方式
     Customer customer=(Customer)session.createQuery("from Customer c order by c.name asc").setMaxResults(1).uniqueResult();
     //采用QBC檢索方式
     Customer customer=(Customer)session.createCriteria(Customer.class).add(Order.asc("name")).setMaxResults(1).uniqueResult();

    posted @ 2007-10-29 16:04 空杯 閱讀(1017) | 評論 (0)編輯 收藏

    MiddlegenIDE的使用

    最近有朋友發(fā)信,說MiddlegenIDE的主頁出了點狀況,登陸上去顯示的內(nèi)容莫名其妙,給新手使用MiddlegenIDE帶來了困難。本座去看了一下,果然是莫名其妙得厲害。下面總結(jié)一個MiddlegenIDE的使用教程吧。

    MiddleGen for Hibernate加上Hibernate_Extension工具包,其實就是用來方便我們從先有的數(shù)據(jù)庫導(dǎo)出表結(jié)構(gòu),生成對應(yīng)的hbm、cfg文件與POJO類代碼。MiddleGenIDE則是MiddleGen的Eclipse插件。整套東西已經(jīng)有很久沒有更新過了。所以雖然本座現(xiàn)在用的eclipse版本3.1.x也能與這個插件正常配合,但是它沒有提供在線update的功能。需要你先下載middlegenide,安裝插件之后重啟Eclipse。如果新開啟的eclipse沒有變化,估計你得用"- clean"參數(shù)再重啟一下,或者去刪除configuration文件夾下面對應(yīng)的文件。

    OK,下面講講怎么去用。首先當然要搭建一個環(huán)境(我現(xiàn)在把MySQL更新到了5.0,Hibernate和當時一樣是2.1),然后我們建一個表,裝一點數(shù)據(jù)。MySQL下面建表的腳本如下:
    1
    2
    3
    4
    5
    6
    DROP TABLE IF EXISTS `test`.`t_user`;
              CREATE TABLE  `test`.`t_user` (
              `id` int(11) NOT NULL auto_increment,
              `name` varchar(255) default NULL,
              PRIMARY KEY  (`id`)
             ) ENGINE=InnoDB DEFAULT CHARSET=gbk;


    有了表之后,建立一個Java工程。然后在Src文件夾上面右鍵,選擇“New-Other”



    然后會彈出Middelgen Biuld File的選項。因為MiddleGen是一個使用Ant編譯文件來調(diào)出IDE的包,所以這個插件實際上就是方便我們用完型填空的方式來編寫這個Biuld File而已。



    點擊后出現(xiàn)下面的界面,對應(yīng)的內(nèi)容一目了然,按照自己的需要填寫即可。



    下面這張圖是填好后的樣子。和本座一樣沒有使用Hibernate 3的朋友,注意在畫了紅圈的Option選項中把hibernate的版本調(diào)低。不然生產(chǎn)的配置文件,會有一個非常詭異的"xml parser無法解析dtd"的錯誤。



    填好所有的東西就點擊Next進入MiddleGen界面,幾乎不用改任何東西直接點擊Generate就能得到配置文件和POJO類了。



    另外,如果你需要在MiddleGen中生成的POJO直接帶Xdoclet的標記的話,需要在找到它自帶的模板文件:
    1
    eclipse\plugins\org.ultimania.middlegenide_1.3.2\resource\template\build-hibernate.xml.vm


    然后把下面這行生成xdoclet tag的值改成true:
    1
    <property name="gen.xdoclet-tag"      value="true">

    posted @ 2007-10-29 16:03 空杯 閱讀(2072) | 評論 (0)編輯 收藏

    Open Session in Test 及自動Rollback

    from: http://m.tkk7.com/rain1102/articles/117541.html

    又是來自Spring這個神奇國度的東西, 你可以讓testCase繼承于AbstractTransactionalDataSourceSpringContextTests,就可以做到Open Session in Test ,解決Hibernate的lazy-load問題;而且接管原來的DAO里的事務(wù)控制定義,通過setDefaultRollback(boolean)方法控制最后回滾還是提交,如果默認為回滾,則測試產(chǎn)生數(shù)據(jù)變動不會影響數(shù)據(jù)庫內(nèi)數(shù)據(jù)。
     
    如果不能繼承于這個基類,可以自己簡單編寫,代碼是這樣的:
       protected PlatformTransactionManager transactionManager;
       protected TransactionStatus transactionStatus;
       protected boolean defaultRollback = true;
       public void setUp()
       {
            transactionManager = (PlatformTransactionManager) ctx.getBean("transactionManager");
            transactionStatus = transactionManager.getTransaction(new DefaultTransactionDefinition());
       }
       public void tearDown()
       {
            if (defaultRollback)
                transactionManager.rollback(this.transactionStatus);
            else
               transactionManager.commit(this.transactionStatus);
        }
    (注,hibernate太奸詐了,如果全部默認回滾,只會在session里干活,一點不寫數(shù)據(jù)庫,達不到完全的測試效果。)

    posted @ 2007-10-29 16:00 空杯 閱讀(282) | 評論 (0)編輯 收藏

    ServletConfig與ServletContext的區(qū)別

    from: http://m.tkk7.com/software5168/archive/2006/09/05/67752.html

    HttpServletRequest,HttpServletResponse:這兩個屬性的作用范圍最小。
        時間上:只是本身請求和應(yīng)答完成就失效,當然轉(zhuǎn)發(fā)是把當前的request對象取出來傳給另一
              個資源,其實本身的request對象還是只生存到本次請求結(jié)束,response也同樣。
        空間上:只能發(fā)送請求的客戶端有效。

        HttpSession:一次連結(jié)到客戶端關(guān)閉,時間作用范圍比上面兩個大,空間任用范圍相同。

        ServletConfig:從一個servlet被實例化后,對任何客戶端在任何時候訪問有效,但僅對本servlet
        有效,一個servlet的ServletConfig對象不能被另一個servlet訪問。

        ServletContext:對任何servlet,任何人在任何時間都有效,這才是真正全局的對象。

        那么,ServletConfig參數(shù)和ServletContext參數(shù)到底應(yīng)該如何使用,如何取得?

        一般來說,對整個應(yīng)用的配置,為了不使用“硬編碼”,應(yīng)該配置為ServletContext參數(shù),比如字
        符集設(shè)定。
        <web-app>
            .................
            <init-param>
                <param-name>charset</param-name> 
                <param-value>GB2312</param-value> 
            </init-param>
            .................
        </web-app>
        注意以上格式只是2。0以后的標準格式,舊容器(引擎)采用服務(wù)商自己的格式配置。注意它的
        父元素應(yīng)該是<web-app>也就是說它是對一個應(yīng)用作用的。

        而如果只有一個特定的servlet要設(shè)定的參數(shù),其它servlet不能共享,應(yīng)該配置為ServletConfig
        參數(shù),如一個讀取附件的servlet要用到絕對目錄,而別的servlet不會用到:
        <servlet>
                <servlet-name>GetAtt</servlet-name>
            <servlet-class>mail.GetAttServlet</servlet-class>
            <init-param>
                <param-name>absPath</param-name> 
                <param-value>/usr/mail/ax/axman/Maildir/</param-value> 
            </init-param>
        </servlet>
        不用說,因為在<servlet>標簽中已經(jīng)指定了name和class,也就是說只有mail.GetAttServlet這個
        servlet中才能取到path,而別的Servlet是不能取到的。

        那么如何訪問這兩個對象的參數(shù)呢?
        訪問ServletConfig參數(shù):
            首先要取得ServletConfig對象,然后調(diào)用它的getInitParameter();方法。要訪問
        ServletConfig對象,jsp中直接使用config內(nèi)置對象,但因為你的JSP編譯后的servlet一般不會被
        加到web.xml中的,所以一般不會通過jsp來取對本JSP編譯后的servlet的配置參數(shù),那么在servlet
        中要得到ServletConfig對象有兩種方法:

        在inii()方法中取到:通過init的重載方法傳遞

        .....
        public class Test extends HttpServlet 
        {
            ServletConfig config;
            public void init(ServletConfig config) throws ServletException {
                this.config = config;
            }
            ..................
        }
        然后在下面的方法中就可以訪問config對象。但要注意,為了確保能從構(gòu)造方法中到到當前servlet的
        config對象,應(yīng)該調(diào)用父類的構(gòu)造方法:
        .....
        public class Test extends HttpServlet 
        {
            ServletConfig config;
            public void init(ServletConfig config) throws ServletException {
                super.init(config);
                this.config = config;
            }
            ..................
        }

        通過getServletConfig()方法直接到時,這樣做的好處是不必調(diào)手工傳遞屬性,想在任何時候都可
        以得到。

        還有第三種方法,要自己實現(xiàn)一些接口,這里作為一般討論就不介紹了。

        要訪問ServletContext對象,只要從現(xiàn)有的ServletConfig對象getServletContext()就可以了,然后
        調(diào)用它的getInitParameter()方法就可以獲取它的參數(shù)。

        按說:ServletContext對象的作用域比ServletConfig作用域大,為什么要從ServletConfig中到得
        ServletContext對象呢?我個人認為:容器保存了很多個ServletContext對象,請求時容器到底取哪一個
        給你呢?那就取其中包含ServletConfig信息的那個給你,就是說取ServletConfig對象的父級對象。就好
        象HttpSession要從requset中取得一樣,就是取那個包含當前requese對象的session對象給你,這只是我
        的個人想法,還沒有來得及看具體實現(xiàn)。反正就這么用吧。

    posted @ 2007-10-29 15:59 空杯 閱讀(2716) | 評論 (0)編輯 收藏

    Struts 1.2 的 HTML 標簽嵌套屬性(如user.name)如何加入 JavaScript 表單驗證

     

    Struts 里面的 之類的表單域標簽, 都有一個屬性叫: property, 這個屬性根據(jù)文檔是可以嵌套的, 例如下面的 formBean 嵌套了一個屬性 User:

    public class User {
    private String username;
    public String getUsername() {
    return username;
    }
    public void setUsername(String username) {
    this.username = username;
    }
    }

     

    public class UserForm extends FormBean {
    private User user;
    public User getUser() {
    return user;
    }
    public void setUser(User user) {
    this.user = user;
    }
    }

     

    那么對應(yīng)的 JSP 頁面里的  Tag 可以寫成:

    <html:text property="user.username" />

    但是如果這時候有人還想給生成的表單加入 JavaScript 驗證的話, 必須用下面的寫法才能通過:

    這是因為 Struts 標簽最后產(chǎn)生的 HTML 如下所示:

     

    <form onsubmit="return validateForm(this);">
    <input name="user.username" />
    form>

     

    直接引用 元素.user.username 肯定會出錯的. 正確的方法參考上上面的代碼段即可.

    posted @ 2007-10-29 15:57 空杯 閱讀(1064) | 評論 (1)編輯 收藏

    FormBean 與 POJO 的集成

     
    代碼
    <form-bean name="loginForm" type="org.apache.struts.validator.DynaValidatorForm">

    <form-property name="user" type="org.layout.model.User"/>

    </form-bean>

    在校驗文件validation.xml中:

    代碼
    <form name="loginForm">
    <field property="user.name" depends="required,minlength,maxlength">
    <arg0 key="label.username"/>
    <arg1 key="${var:minlength}" name="minlength" resource="false"/>
    <arg2 key="${var:maxlength}" name="maxlength" resource="false"/>
    <var>
    <var-name>maxlength</var-name>
    <var-value>16</var-value>
    </var>
    <var>
    <var-name>minlength</var-name>
    <var-value>2</var-value>
    </var>
    </field>
    <field property="user.password" 
    depends
    ="required,minlength,maxlength">
    <arg0 key="label.password"/>
    <arg1 key="${var:minlength}" name="minlength" resource="false"/>
    <arg2 key="${var:maxlength}" name="maxlength" resource="false"/>
    <var>
    <var-name>maxlength</var-name>
    <var-value>16</var-value>
    </var>
    <var>
    <var-name>minlength</var-name>
    <var-value>2</var-value>
    </var>
    </field>
    </form>
    在action中如下調(diào)用:
    代碼
            DynaActionForm aForm = (DynaActionForm)form;
            ActionMessages messages 
    = form.validate(mapping, request);
            
    if(!messages.isEmpty())
            
    {
                saveMessages(request, messages);
                
    return mapping.findForward("prompt");
            }

            User user 
    = (User)aForm.get("user");
            UserDAO userDAO 
    = (UserDAO)getWebApplicationContext().getBean("userDAO");
            user 
    = userDAO.get(user.getName(), user.getPassword());
    其實這樣子的集成感覺很不錯的,除了少了dto和轉(zhuǎn)換(至少你要用一個copyProperty吧),pojo實現(xiàn)了序列化,
    甚至可以直接深入到Hibernate底層,這樣子就省缺了以前一直討論formbean和po是不是合并或者在哪個層面上進
    行轉(zhuǎn)換。

    posted @ 2007-10-29 15:54 空杯 閱讀(327) | 評論 (0)編輯 收藏

    主站蜘蛛池模板: 黄色一级毛片免费| 无码国产精品一区二区免费虚拟VR| 亚洲精品美女久久久久99| 日韩免费在线观看视频| 中文字幕亚洲综合小综合在线| 日韩精品电影一区亚洲| 91视频免费网址| 成人婷婷网色偷偷亚洲男人的天堂 | 免费v片在线观看| 无码国产精品一区二区免费式芒果| 亚洲人成77777在线观看网| 亚洲毛片av日韩av无码| 国产免费的野战视频| 国产国产人免费人成成免视频| 亚洲理论片在线观看| 亚洲真人日本在线| 免费精品人在线二线三线区别 | 国产亚洲福利一区二区免费看| 成人av片无码免费天天看| 亚洲熟女www一区二区三区| 亚洲精品无码不卡在线播HE| 在线播放高清国语自产拍免费| a毛片免费播放全部完整| 亚洲aⅴ无码专区在线观看春色 | 免费毛片在线播放| 99爱在线观看免费完整版| 无遮挡a级毛片免费看| 亚洲av永久无码嘿嘿嘿| 亚洲第一AV网站| 亚洲国产黄在线观看| 亚洲日韩在线观看免费视频| 57pao一国产成视频永久免费 | 免费能直接在线观看黄的视频 | 水蜜桃视频在线观看免费播放高清| 亚洲午夜成人精品无码色欲| 久久水蜜桃亚洲av无码精品麻豆| 中文字幕亚洲第一| 亚洲av中文无码| 日本高清免费不卡视频| 成人毛片手机版免费看| 国产黄色免费网站|