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

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

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

    抽象類與接口的區別(轉載)

    abstract?class和interface是Java語言中對于抽象類定義進行支持的兩種機制,正是由于這兩種機制的存在,才賦予了Java強大的面向對象能力。?abstract?class和interface之間在對于抽象類定義的支持方面具有很大的相似性,甚至可以相互替換,因此很多開發者在進行抽象類定義時對于?abstract?class和interface的選擇顯得比較隨意。
    其實,兩者之間還是有很大的區別的,對于它們的選擇甚至反映出對于問題領域本質的理解、對于設計意圖的理解是否正確、合理。本文將對它們之間的區別進行一番剖析,試圖給開發者提供一個在二者之間進行選擇的依據。
    一、理解抽象類
    abstract?class和interface在Java語言中都是用來進行抽象類(本文中的抽象類并非從abstract?class翻譯而來,它表示的是一個抽象體,而abstract?class為Java語言中用于定義抽象類的一種方法,請讀者注意區分)定義的,那么什么是抽象類,使用抽象類能為我們帶來什么好處呢?
    在面向對象的概念中,我們知道所有的對象都是通過類來描繪的,但是反過來卻不是這樣。并不是所有的類都是用來描繪對象的,如果一個類中沒有包含足夠的信息來描繪一個具體的對象,這樣的類就是抽象類。抽象類往往用來表征我們在對問題領域進行分析、設計中得出的抽象概念,是對一系列看上去不同,但是本質上相同的具體概念的抽象。
    比如:如果我們進行一個圖形編輯軟件的開發,就會發現問題領域存在著圓、三角形這樣一些具體概念,它們是不同的,但是它們又都屬于形狀這樣一個概念,形狀這個概念在問題領域是不存在的,它就是一個抽象概念。正是因為抽象的概念在問題領域沒有對應的具體概念,所以用以表征抽象概念的抽象類是不能夠實例化的。
    在面向對象領域,抽象類主要用來進行類型隱藏。我們可以構造出一個固定的一組行為的抽象描述,但是這組行為卻能夠有任意個可能的具體實現方式。這個抽象描述就是抽象類,而這一組任意個可能的具體實現則表現為所有可能的派生類。模塊可以操作一個抽象體。由于模塊依賴于一個固定的抽象體,因此它可以是不允許修改的;同時,通過從這個抽象體派生,也可擴展此模塊的行為功能。熟悉OCP的讀者一定知道,為了能夠實現面向對象設計的一個最核心的原則OCP(Open-Closed?Principle),抽象類是其中的關鍵所在。
    二、從語法定義層面看abstract?class和interface
    在語法層面,Java語言對于abstract?class和interface給出了不同的定義方式,下面以定義一個名為Demo的抽象類為例來說明這種不同。使用abstract?class的方式定義Demo抽象類的方式如下:

    abstract?class?Demo?{
    abstract?void?method1();
    abstract?void?method2();




    使用interface的方式定義Demo抽象類的方式如下:

    interface?Demo?{
    void?method1();
    void?method2();

    }


    在abstract?class方式中,Demo可以有自己的數據成員,也可以有非abstarct的成員方法,而在interface方式的實現中,Demo只能夠有靜態的不能被修改的數據成員(也就是必須是static?final的,不過在interface中一般不定義數據成員),所有的成員方法都是abstract的。從某種意義上說,interface是一種特殊形式的abstract?class。
    從編程的角度來看,abstract?class和interface都可以用來實現"design?by?contract"的思想。但是在具體的使用上面還是有一些區別的。
    首先,abstract?class在Java語言中表示的是一種繼承關系,一個類只能使用一次繼承關系。但是,一個類卻可以實現多個interface。也許,這是Java語言的設計者在考慮Java對于多重繼承的支持方面的一種折中考慮吧。
    其次,在abstract?class的定義中,我們可以賦予方法的默認行為。但是在interface的定義中,方法卻不能擁有默認行為,為了繞過這個限制,必須使用委托,但是這會?增加一些復雜性,有時會造成很大的麻煩。
    在抽象類中不能定義默認行為還存在另一個比較嚴重的問題,那就是可能會造成維護上的麻煩。因為如果后來想修改類的界面(一般通過abstract?class或者interface來表示)以適應新的情況(比如,添加新的方法或者給已用的方法中添加新的參數)時,就會非常的麻煩,可能要花費很多的時間(對于派生類很多的情況,尤為如此)。但是如果界面是通過abstract?class來實現的,那么可能就只需要修改定義在abstract?class中的默認行為就可以了。
    同樣,如果不能在抽象類中定義默認行為,就會導致同樣的方法實現出現在該抽象類的每一個派生類中,違反了?"one?rule,one?place"原則,造成代碼重復,同樣不利于以后的維護。因此,在abstract?class和interface間進行選擇時要非常的小心。
    三、從設計理念層面看abstract?class和interface
    上面主要從語法定義和編程的角度論述了abstract?class和interface的區別,這些層面的區別是比較低層次的、非本質的。本文將從另一個層面:abstract?class和interface所反映出的設計理念,來分析一下二者的區別。作者認為,從這個層面進行分析才能理解二者概念的本質所在。
    前面已經提到過,abstarct?class在Java語言中體現了一種繼承關系,要想使得繼承關系合理,父類和派生類之間必須存在"is?a"關系,即父類和派生類在概念本質上應該是相同的。對于interface?來說則不然,并不要求interface的實現者和interface定義在概念本質上是一致的,僅僅是實現了interface定義的契約而已。為了使論述便于理解,下面將通過一個簡單的實例進行說明。
    考慮這樣一個例子,假設在我們的問題領域中有一個關于Door的抽象概念,該Door具有執行兩個動作open和close,此時我們可以通過abstract?class或者interface來定義一個表示該抽象概念的類型,定義方式分別如下所示:

    使用abstract?class方式定義Door:

    abstract?class?Door?{
    abstract?void?open();
    abstract?void?close();
    }

    使用interface方式定義Door:

    interface?Door?{
    void?open();
    void?close();
    }


    其他具體的Door類型可以extends使用abstract?class方式定義的Door或者implements使用interface方式定義的Door。看起來好像使用abstract?class和interface沒有大的區別。
    如果現在要求Door還要具有報警的功能。我們該如何設計針對該例子的類結構呢(在本例中,主要是為了展示abstract?class和interface反映在設計理念上的區別,其他方面無關的問題都做了簡化或者忽略)下面將羅列出可能的解決方案,并從設計理念層面對這些不同的方案進行分析。
    解決方案一:
    簡單的在Door的定義中增加一個alarm方法,如下:

    abstract?class?Door?{
    abstract?void?open();
    abstract?void?close();
    abstract?void?alarm();
    }


    或者

    interface?Door?{
    void?open();
    void?close();
    void?alarm();
    }


    那么具有報警功能的AlarmDoor的定義方式如下:

    class?AlarmDoor?extends?Door?{
    void?open()?{?…?}
    void?close()?{?…?}
    void?alarm()?{?…?}
    }


    或者

    class?AlarmDoor?implements?Door?{
    void?open()?{?…?}
    void?close()?{?…?}
    void?alarm()?{?…?}



    這種方法違反了面向對象設計中的一個核心原則ISP(Interface?Segregation?Priciple),在Door的定義中把Door概念本身固有的行為方法和另外一個概念"報警器"的行為方法混在了一起。這樣引起的一個問題是那些僅僅依賴于Door這個概念的模塊會因為"報警器"這個概念的改變(比如:修改alarm方法的參數)而改變,反之依然。
    解決方案二:
    既然open、close和alarm屬于兩個不同的概念,根據ISP原則應該把它們分別定義在代表這兩個概念的抽象類中。定義方式有:這兩個概念都使用?abstract?class方式定義;兩個概念都使用interface方式定義;一個概念使用abstract?class方式定義,另一個概念使用interface方式定義。
    顯然,由于Java語言不支持多重繼承,所以兩個概念都使用abstract?class方式定義是不可行的。后面兩種方式都是可行的,但是對于它們的選擇卻反映出對于問題領域中的概念本質的理解、對于設計意圖的反映是否正確、合理。我們一一來分析、說明。
    如果兩個概念都使用interface方式來定義,那么就反映出兩個問題:
    1、我們可能沒有理解清楚問題領域,AlarmDoor在概念本質上到底是Door還是報警器?
    2、如果我們對于問題領域的理解沒有問題,比如:我們通過對于問題領域的分析發現AlarmDoor在概念本質上和Door是一致的,那么我們在實現時就沒有能夠正確的揭示我們的設計意圖,因為在這兩個概念的定義上(均使用interface方式定義)反映不出上述含義。
    如果我們對于問題領域的理解是:AlarmDoor在概念本質上是Door,同時它有具有報警的功能。我們該如何來設計、實現來明確的反映出我們的意思呢?前面已經說過,?abstract?class在Java語言中表示一種繼承關系,而繼承關系在本質上是"is?a"關系。所以對于Door這個概念,我們應該使用abstarct?class方式來定義。另外,AlarmDoor又具有報警功能,說明它又能夠完成報警概念中定義的行為,所以報警概念可以通過interface方式定義。如下所示:

    abstract?class?Door?{
    abstract?void?open();
    abstract?void?close();
    }
    interface?Alarm?{
    void?alarm();
    }
    class?AlarmDoor?extends?Door?implements?Alarm?{
    void?open()?{?…?}
    void?close()?{?…?}
    void?alarm()?{?…?}
    }


    這種實現方式基本上能夠明確的反映出我們對于問題領域的理解,正確的揭示我們的設計意圖。其實abstract?class表示的是"is?a"關系,interface表示的是"like?a"關系,大家在選擇時可以作為一個依據,當然這是建立在對問題領域的理解上的,比如:如果我們認為AlarmDoor在概念本質上是報警器,同時又具有?Door的功能,那么上述的定義方式就要反過來了。
    abstract?class和interface是Java語言中的兩種定義抽象類的方式,它們之間有很大的相似性。但是對于它們的選擇卻又往往反映出對于問題領域中的概念本質的理解、對于設計意圖的反映是否正確、合理,因為它們表現了概念間的不同的關系(雖然都能夠實現需求的功能)。這其實也是語言的一種的慣用法,希望讀者朋友能夠細細體會

    posted @ 2006-06-09 11:31 nbt 閱讀(187) | 評論 (0)編輯 收藏

    徹底明白Java的IO系統(轉載)

    一.?Input和Output
    1.?stream代表的是任何有能力產出數據的數據源,或是任何有能力接收數據的接收源。在Java的IO中,所有的stream(包括Input和Out?stream)都包括兩種類型:
    1.1?以字節為導向的stream
    以字節為導向的stream,表示以字節為單位從stream中讀取或往stream中寫入信息。以字節為導向的stream包括下面幾種類型:
    1)?input stream:
    1)?ByteArrayInputStream:把內存中的一個緩沖區作為InputStream使用
    2)?StringBufferInputStream:把一個String對象作為InputStream
    3)?FileInputStream:把一個文件作為InputStream,實現對文件的讀取操作
    4)?PipedInputStream:實現了pipe的概念,主要在線程中使用
    5)?SequenceInputStream:把多個InputStream合并為一個InputStream
    2)?Out stream
    1)?ByteArrayOutputStream:把信息存入內存中的一個緩沖區中
    2)?FileOutputStream:把信息存入文件中
    3)?PipedOutputStream:實現了pipe的概念,主要在線程中使用
    4)?SequenceOutputStream:把多個OutStream合并為一個OutStream
    1.2?以Unicode字符為導向的stream
    以Unicode字符為導向的stream,表示以Unicode字符為單位從stream中讀取或往stream中寫入信息。以Unicode字符為導向的stream包括下面幾種類型:
    1)?Input Stream
    1)?CharArrayReader:與ByteArrayInputStream對應
    2)?StringReader:與StringBufferInputStream對應
    3)?FileReader:與FileInputStream對應
    4)?PipedReader:與PipedInputStream對應
    2)?Out Stream
    1)?CharArrayWrite:與ByteArrayOutputStream對應
    2)?StringWrite:無與之對應的以字節為導向的stream
    3)?FileWrite:與FileOutputStream對應
    4)?PipedWrite:與PipedOutputStream對應

    以字符為導向的stream基本上對有與之相對應的以字節為導向的stream。兩個對應類實現的功能相同,字是在操作時的導向不同。如CharArrayReader:和ByteArrayInputStream的作用都是把內存中的一個緩沖區作為InputStream使用,所不同的是前者每次從內存中讀取一個字節的信息,而后者每次從內存中讀取一個字符。
    1.3?兩種不現導向的stream之間的轉換
    InputStreamReader和OutputStreamReader:把一個以字節為導向的stream轉換成一個以字符為導向的stream。
    2.?stream添加屬性
    2.1?“為stream添加屬性”的作用
    運用上面介紹的Java中操作IO的API,我們就可完成我們想完成的任何操作了。但通過FilterInputStream和FilterOutStream的子類,我們可以為stream添加屬性。下面以一個例子來說明這種功能的作用。
    如果我們要往一個文件中寫入數據,我們可以這樣操作:
    FileOutStream?fs?=?new?FileOutStream(“test.txt”);
    然后就可以通過產生的fs對象調用write()函數來往test.txt文件中寫入數據了。但是,如果我們想實現“先把要寫入文件的數據先緩存到內存中,再把緩存中的數據寫入文件中”的功能時,上面的API就沒有一個能滿足我們的需求了。但是通過FilterInputStream和FilterOutStream的子類,為FileOutStream添加我們所需要的功能。
    2.2?FilterInputStream的各種類型
    2.2.1?用于封裝以字節為導向的InputStream
    1)?DataInputStream:從stream中讀取基本類型(int、char等)數據。
    2)?BufferedInputStream:使用緩沖區
    3)?LineNumberInputStream:會記錄input?stream內的行數,然后可以調用getLineNumber()和setLineNumber(int)
    4)?PushbackInputStream:很少用到,一般用于編譯器開發
    2.2.2?用于封裝以字符為導向的InputStream
    1)?沒有與DataInputStream對應的類。除非在要使用readLine()時改用BufferedReader,否則使用DataInputStream
    2)?BufferedReader:與BufferedInputStream對應
    3)?LineNumberReader:與LineNumberInputStream對應
    4)?PushBackReader:與PushbackInputStream對應
    2.3?FilterOutStream的各種類型
    2.2.3?用于封裝以字節為導向的OutputStream
    1)?DataIOutStream:往stream中輸出基本類型(int、char等)數據。
    2)?BufferedOutStream:使用緩沖區
    3)?PrintStream:產生格式化輸出
    2.2.4?用于封裝以字符為導向的OutputStream
    1)?BufferedWrite:與對應
    2)?PrintWrite:與對應
    3.?RandomAccessFile
    1)?可通過RandomAccessFile對象完成對文件的讀寫操作
    2)?在產生一個對象時,可指明要打開的文件的性質:r,只讀;w,只寫;rw可讀寫
    3)?可以直接跳到文件中指定的位置
    4.?I/O應用的一個例子

    import?java.io.*;
    public?class?TestIO{
    public?static?void?main(String[]?args)
    throws?IOException{
    //1.以行為單位從一個文件讀取數據
    BufferedReader?in?=?
    new?BufferedReader(
    new?FileReader("F:\\nepalon\\TestIO.java"));
    String?s,?s2?=?new?String();
    while((s?=?in.readLine())?!=?null)
    s2?+=?s?+?"\n";
    in.close();

    //1b.?接收鍵盤的輸入
    BufferedReader?stdin?=?
    new?BufferedReader(
    new?InputStreamReader(System.in));
    System.out.println("Enter?a?line:");
    System.out.println(stdin.readLine());

    //2.?從一個String對象中讀取數據
    StringReader?in2?=?new?StringReader(s2);
    int?c;
    while((c?=?in2.read())?!=?-1)
    System.out.println((char)c);
    in2.close();

    //3.?從內存取出格式化輸入
    try{
    DataInputStream?in3?=?
    new?DataInputStream(
    new?ByteArrayInputStream(s2.getBytes()));
    while(true)
    System.out.println((char)in3.readByte());?
    }
    catch(EOFException?e){
    System.out.println("End?of?stream");
    }

    //4.?輸出到文件
    try{
    BufferedReader?in4?=
    new?BufferedReader(
    new?StringReader(s2));
    PrintWriter?out1?=
    new?PrintWriter(
    new?BufferedWriter(
    new?FileWriter("F:\\nepalon\\?TestIO.out")));
    int?lineCount?=?1;
    while((s?=?in4.readLine())?!=?null)
    out1.println(lineCount++?+?":"?+?s);
    out1.close();
    in4.close();
    }
    catch(EOFException?ex){
    System.out.println("End?of?stream");
    }

    //5.?數據的存儲和恢復
    try{
    DataOutputStream?out2?=?
    new?DataOutputStream(
    new?BufferedOutputStream(
    new?FileOutputStream("F:\\nepalon\\?Data.txt")));
    out2.writeDouble(3.1415926);
    out2.writeChars("\nThas?was?pi:writeChars\n");
    out2.writeBytes("Thas?was?pi:writeByte\n");
    out2.close();
    DataInputStream?in5?=
    new?DataInputStream(
    new?BufferedInputStream(
    new?FileInputStream("F:\\nepalon\\?Data.txt")));
    BufferedReader?in5br?=
    new?BufferedReader(
    new?InputStreamReader(in5));
    System.out.println(in5.readDouble());
    System.out.println(in5br.readLine());
    System.out.println(in5br.readLine());
    }

    catch(EOFException?e){
    System.out.println("End?of?stream");
    }

    //6.?通過RandomAccessFile操作文件
    RandomAccessFile?rf?=
    new?RandomAccessFile("F:\\nepalon\\?rtest.dat",?"rw");
    for(int?i=0;?i<10;?i++)
    rf.writeDouble(i*1.414);
    rf.close();

    rf?=?new?RandomAccessFile("F:\\nepalon\\?rtest.dat",?"r");
    for(int?i=0;?i<10;?i++)
    System.out.println("Value?"?+?i?+?":"?+?rf.readDouble());
    rf.close();

    rf?=?new?RandomAccessFile("F:\\nepalon\\?rtest.dat",?"rw");
    rf.seek(5*8);
    rf.writeDouble(47.0001);
    rf.close();

    rf?=?new?RandomAccessFile("F:\\nepalon\\?rtest.dat",?"r");
    for(int?i=0;?i<10;?i++)
    System.out.println("Value?"?+?i?+?":"?+?rf.readDouble());
    rf.close();
    }
    }
    關于代碼的解釋(以區為單位):
    1區中,當讀取文件時,先把文件內容讀到緩存中,當調用in.readLine()時,再從緩存中以字符的方式讀取數據(以下簡稱“緩存字節讀取方式”)。
    1b區中,由于想以緩存字節讀取方式從標準IO(鍵盤)中讀取數據,所以要先把標準IO(System.in)轉換成字符導向的stream,再進行BufferedReader封裝。
    2區中,要以字符的形式從一個String對象中讀取數據,所以要產生一個StringReader類型的stream。
    4區中,對String對象s2讀取數據時,先把對象中的數據存入緩存中,再從緩沖中進行讀取;對TestIO.out文件進行操作時,先把格式化后的信息輸出到緩存中,再把緩存中的信息輸出到文件中。
    5區中,對Data.txt文件進行輸出時,是先把基本類型的數據輸出屋緩存中,再把緩存中的數據輸出到文件中;對文件進行讀取操作時,先把文件中的數據讀取到緩存中,再從緩存中以基本類型的形式進行讀取。注意in5.readDouble()這一行。因為寫入第一個writeDouble(),所以為了正確顯示。也要以基本類型的形式進行讀取。
    6區是通過RandomAccessFile類對文件進行操作。

    posted @ 2006-06-09 10:30 nbt 閱讀(215) | 評論 (0)編輯 收藏

    JavaMail API詳解(轉自duduwjf )

    摘要:
    JavaMail?API是讀取、撰寫、發送電子信息的可選包。我們可用它來建立如Eudora、Foxmail、MS?Outlook?Express一般的郵件用戶代理程序(Mail?User?Agent,簡稱MUA)。讓我們看看JavaMail?API是如何提供信息訪問功能的吧!JavaMail?API被設計用于以不依賴協議的方式去發送和接收電子信息,文中著重:如何以不依賴于協議的方式發送接收電子信息,這也是本文所要描述的.?文章工具
    收藏
    投票評分
    發表評論
    復制鏈接
    版權聲明:本文可以自由轉載,轉載時請務必以超鏈接形式標明文章原始出處和作者信息及本聲明
    作者:cleverpig(作者的Blog:http://blog.matrix.org.cn/page/cleverpig)
    原文:http://www.matrix.org.cn/resource/article/44/44101_JavaMail.html
    關鍵字:java,mail,pop,smtp

    一、JavaMail?API簡介
    JavaMail?API是讀取、撰寫、發送電子信息的可選包。我們可用它來建立如Eudora、Foxmail、MS?Outlook?Express一般的郵件用戶代理程序(Mail?User?Agent,簡稱MUA)。而不是像sendmail或者其它的郵件傳輸代理(Mail?Transfer?Agent,簡稱MTA)程序那樣可以傳送、遞送、轉發郵件。從另外一個角度來看,我們這些電子郵件用戶日常用MUA程序來讀寫郵件,而MUA依賴著MTA處理郵件的遞送。
    在清楚了到MUA與MTA之間的關系后,讓我們看看JavaMail?API是如何提供信息訪問功能的吧!JavaMail?API被設計用于以不依賴協議的方式去發送和接收電子信息,這個API被分為兩大部分:

    基本功能:如何以不依賴于協議的方式發送接收電子信息,這也是本文所要描述的,不過在下文中,大家將看到這只是一廂情愿而已。
    第二個部分則是依賴特定協議的,比如SMTP、POP、IMAP、NNTP協議。在這部分的JavaMail?API是為了和服務器通訊,并不在本文的內容中。

    二、相關協議一覽
    在我們步入JavaMail?API之前,先看一下API所涉及的協議。以下便是大家日常所知、所樂于使用的4大信息傳輸協議:
    SMTP
    POP
    IMAP
    MIME
    當然,上面的4個協議,并不是全部,還有NNTP和其它一些協議可用于傳輸信息,但是由于不常用到,所以本文便不提及了。理解這4個基本的協議有助于我們更好的使用JavaMail?API。然而JavaMail?API是被設計為與協議無關的,目前我們并不能克服這些協議的束縛。確切的說,如果我們使用的功能并不被我們選擇的協議支持,那么JavaMail?API并不可能如魔術師一樣神奇的賦予我們這種能力。

    1.SMTP
    簡單郵件傳輸協議定義了遞送郵件的機制。在下文中,我們將使用基于Java-Mail的程序與公司或者ISP的SMTP服務器進行通訊。這個SMTP服務器將郵件轉發到接收者的SMTP服務器,直至最后被接收者通過POP或者IMAP協議獲取。這并不需要SMTP服務器使用支持授權的郵件轉發,但是卻的確要注意SMTP服務器的正確設置(SMTP服務器的設置與JavaMail?API無關)。

    2.POP
    POP是一種郵局協議,目前為第3個版本,即眾所周知的POP3。POP定義了一種用戶如何獲得郵件的機制。它規定了每個用戶使用一個單獨的郵箱。大多數人在使用POP時所熟悉的功能并非都被支持,例如查看郵箱中的新郵件數量。而這個功能是微軟的Outlook內建的,那么就說明微軟Outlook之類的郵件客戶端軟件是通過查詢最近收到的郵件來計算新郵件的數量來實現前面所說的功能。因此在我們使用JavaMail?API時需要注意,當需要獲得如前面所講的新郵件數量之類的信息時,我們不得不自己進行計算。

    3.IMAP
    IMAP使用在接收信息的高級協議,目前版本為第4版,所以也被稱為IMAP4。需要注意的是在使用IMAP時,郵件服務器必須支持該協議。從這個方面講,我們并不能完全使用IMAP來替代POP,不能期待IMAP在任何地方都被支持。假如郵件服務器支持IMAP,那么我們的郵件程序將能夠具有以下被IMAP所支持的特性:每個用戶在服務器上可具有多個目錄,這些目錄能在多個用戶之間共享。
    其與POP相比高級之處顯而易見,但是在嘗試采取IMAP時,我們認識到它并不是十分完美的:由于IMAP需要從其它服務器上接收新信息,將這些信息遞送給用戶,維護每個用戶的多個目錄,這都為郵件服務器帶來了高負載。并且IMAP與POP的一個不同之處是POP用戶在接收郵件時將從郵件服務器上下載郵件,而IMAP允許用戶直接訪問郵件目錄,所以在郵件服務器進行備份作業時,由于每個長期使用此郵件系統的用戶所用的郵件目錄會占有很大的空間,這將直接導致郵件服務器上磁盤空間暴漲。

    4.MIME
    MIME并不是用于傳送郵件的協議,它作為多用途郵件的擴展定義了郵件內容的格式:信息格式、附件格式等等。一些RFC標準都涉及了MIME:RFC?822,?RFC?2045,?RFC?2046,?RFC?2047,有興趣的Matrixer可以閱讀一下。而作為JavaMail?API的開發者,我們并不需關心這些格式定義,但是這些格式被用在了程序中。

    5.NNTP和其它的第三方協議
    正因為JavaMail?API在設計時考慮到與第三方協議實現提供商之間的分離,故我們可以很容易的添加一些第三方協議。SUN維護著一個第三方協議實現提供商的列表:http://java.sun.com/products/javamail/Third_Party.html,通過此列表我們可以找到所需要的而又不被SUN提供支持的第三方協議:比如NNTP這個新聞組協議和S/MIME這個安全的MIME協議。

    三、安裝
    1.安裝JavaMail
    為了使用JavaMail?API,需要從http://java.sun.com/products/javamail/downloads/index.html下載文件名格式為javamail-[version].zip的文件(這個文件中包括了JavaMail實現),并將其中的mail.jar文件添加到CLASSPATH中。這個實現提供了對SMTP、IMAP4、POP3的支持。
    注意:在安裝JavaMail實現之后,我們將在demo目錄中發現許多有趣的簡單實例程序。
    在安裝了JavaMail之后,我們還需要安裝JavaBeans?Activation?Framework,因為這個框架是JavaMail?API所需要的。如果我們使用J2EE的話,那么我們并無需單獨下載JavaMail,因為它存在于J2EE.jar中,只需將J2EE.jar加入到CLASSPATH即可。

    2.安裝JavaBeans?Activation?Framework
    http://java.sun.com/products/javabeans/glasgow/jaf.html下載JavaBeans?Activation?Framework,并將其添加到CLASSPATH中。此框架增加了對任何數據塊的分類、以及對它們的處理的特性。這些特性是JavaMail?API需要的。雖然聽起來這些特性非常模糊,但是它對于我們的JavaMail?API來說只是提供了基本的MIME類型支持。
    到此為止,我們應當把mail.jar和activation.jar都添加到了CLASSPATH中。
    當然如果從方便的角度講,直接把這兩個Jar文件復制到JRE目錄的lib/ext目錄中也可以。

    四、初次認識JavaMail?API
    1.了解我們的JavaMail環境
    A.縱覽JavaMail核心類結構
    打開JavaMail.jar文件,我們將發現在javax.mail的包下面存在著一些核心類:Session、Message、Address、Authenticator、Transport、Store、Folder。而且在javax.mail.internet包中還有一些常用的子類。
    B.Session
    Session類定義了基本的郵件會話。就像Http會話那樣,我們進行收發郵件的工作都是基于這個會話的。Session對象利用了java.util.Properties對象獲得了郵件服務器、用戶名、密碼信息和整個應用程序都要使用到的共享信息。
    Session類的構造方法是私有的,所以我們可以使用Session類提供的getDefaultInstance()這個靜態工廠方法獲得一個默認的Session對象:
    Properties?props?=?new?Properties();//?fill?props?with?any?informationSession?session?=?Session.getDefaultInstance(props,?null);
    或者使用getInstance()這個靜態工廠方法獲得自定義的Session:?
    Properties?props?=?new?Properties();//?fill?props?with?any?informationSession?session?=?Session.getInstance(props,?null);
    從上面的兩個例子中不難發現,getDefaultInstance()和getInstance()方法的第二個參數都是null,這是因為在上面的例子中并沒有使用到郵件授權,下文中將對授權進行詳細介紹。
    從很多的實例看,在對mail?server進行訪問的過程中使用共享的Session是足夠的,即使是工作在多個用戶郵箱的模式下也不例外。

    C.Message
    當我們建立了Session對象后,便可以被發送的構造信息體了。在這里SUN提供了Message類型來幫助開發者完成這項工作。由于Message是一個抽象類,大多數情況下,我們使用javax.mail.internet.MimeMessage這個子類,該類是使用MIME類型、MIME信息頭的郵箱信息。信息頭只能使用US-ASCII字符,而非ASCII字符將通過編碼轉換為ASCII的方式使用。
    為了建立一個MimeMessage對象,我們必須將Session對象作為MimeMessage構造方法的參數傳入:
    MimeMessage?message?=?new?MimeMessage(session);
    注意:對于MimeMessage類來講存在著多種構造方法,比如使用輸入流作為參數的構造方法。

    在建立了MimeMessage對象后,我們需要設置它的各個part,對于MimeMessage類來說,這些part就是MimePart接口。最基本的設置信息內容的方法就是通過表示信息內容和米么類型的參數調用setContent()方法:
    message.setContent("Hello",?"text/plain");
    然而,如果我們所使用的MimeMessage中信息內容是文本的話,我們便可以直接使用setText()方法來方便的設置文本內容。
    message.setText("Hello");
    前面所講的兩種方法,對于文本信息,后者更為合適。而對于其它的一些信息類型,比如HTML信息,則要使用前者。
    別忘記了,使用setSubject()方法對郵件設置郵件主題:
    message.setSubject("First");

    D.Address
    到這里,我們已經建立了Session和Message,下面將介紹如何使用郵件地址類:Address。像Message一樣,Address類也是一個抽象類,所以我們將使用javax.mail.internet.InternetAddress這個子類。
    通過傳入代表郵件地址的字符串,我們可以建立一個郵件地址類:
    Address?address?=?new?InternetAddress("president@whitehouse.gov");?
    如果要在郵件地址后面增加名字的話,可以通過傳遞兩個參數:代表郵件地址和名字的字符串來建立一個具有郵件地址和名字的郵件地址類:
    Address?address?=?new?InternetAddress("president@whitehouse.gov",?"George?Bush");?
    本文在這里所講的郵件地址類是為了設置郵件信息的發信人和收信人而準備的,在建立了郵件地址類后,我們通過message的setFrom()和setReplyTo()兩種方法設置郵件的發信人:
    message.setFrom(address);message.setReplyTo(address);
    若在郵件中存在多個發信人地址,我們可用addForm()方法增加發信人:
    Address?address[]?=?...;message.addFrom(address);
    為了設置收信人,我們使用addRecipient()方法增加收信人,此方法需要使用Message.RecipientType的常量來區分收信人的類型:
    message.addRecipient(type,?address)
    下面是Message.RecipientType的三個常量:
    Message.RecipientType.TO
    Message.RecipientType.CC
    Message.RecipientType.BCC
    因此,如果我們要發送郵件給總統,并發用一個副本給第一夫人的話,下面的方法將被用到:
    Address?toAddress?=?new?InternetAddress("vice.president@whitehouse.gov");Address?ccAddress?=?new?InternetAddress("first.lady@whitehouse.gov");message.addRecipient(Message.RecipientType.TO,?toAddress);message.addRecipient(Message.RecipientType.CC,?ccAddress);
    JavaMail?API并沒有提供檢查郵件地址有效性的機制。當然我們可以自己完成這個功能:驗證郵件地址的字符是否按照RFC822規定的格式書寫或者通過DNS服務器上的MX記錄驗證等。

    E.Authenticator
    像java.net類那樣,JavaMail?API通過使用授權者類(Authenticator)以用戶名、密碼的方式訪問那些受到保護的資源,在這里“資源”就是指郵件服務器。在javax.mail包中可以找到這個JavaMail的授權者類(Authenticator)。
    在使用Authenticator這個抽象類時,我們必須采用繼承該抽象類的方式,并且該繼承類必須具有返回PasswordAuthentication對象(用于存儲認證時要用到的用戶名、密碼)getPasswordAuthentication()方法。并且要在Session中進行注冊,使Session能夠了解在認證時該使用哪個類。
    下面代碼片斷中的MyAuthenticator就是一個Authenticator的子類。
    Properties?props?=?new?Properties();//?fill?props?with?any?informationAuthenticator?auth?=?new?MyAuthenticator();Session?session?=?Session.getDefaultInstance(props,?auth);

    F.Transport
    在發送信息時,Transport類將被用到。這個類實現了發送信息的協議(通稱為SMTP),此類是一個抽象類,我們可以使用這個類的靜態方法send()來發送消息:
    Transport.send(message);
    當然,方法是多樣的。我們也可由Session獲得相應協議對應的Transport實例。并通過傳遞用戶名、密碼、郵件服務器主機名等參數建立與郵件服務器的連接,并使用sendMessage()方法將信息發送,最后關閉連接:
    message.saveChanges();?//?implicit?with?send()Transport?transport?=?session.getTransport("smtp");transport.connect(host,?username,?password);transport.sendMessage(message,?message.getAllRecipients());transport.close();
    評論:上面的方法是一個很好的方法,尤其是在我們在同一個郵件服務器上發送多個郵件時。因為這時我們將在連接郵件服務器后連續發送郵件,然后再關閉掉連接。send()這個基本的方法是在每次調用時進行與郵件服務器的連接的,對于在同一個郵件服務器上發送多個郵件來講可謂低效的方式。
    注意:如果需要在發送郵件過程中監控mail命令的話,可以在發送前設置debug標志:
    session.setDebug(true)。

    G.Store和Folder
    接收郵件和發送郵件很類似都要用到Session。但是在獲得Session后,我們需要從Session中獲取特定類型的Store,然后連接到Store,這里的Store代表了存儲郵件的郵件服務器。在連接Store的過程中,極有可能需要用到用戶名、密碼或者Authenticator。
    //?Store?store?=?session.getStore("imap");Store?store?=?session.getStore("pop3");store.connect(host,?username,?password);
    在連接到Store后,一個Folder對象即目錄對象將通過Store的getFolder()方法被返回,我們可從這個Folder中讀取郵件信息:
    Folder?folder?=?store.getFolder("INBOX");folder.open(Folder.READ_ONLY);Message?message[]?=?folder.getMessages();
    上面的例子首先從Store中獲得INBOX這個Folder(對于POP3協議只有一個名為INBOX的Folder有效),然后以只讀(Folder.READ_ONLY)的方式打開Folder,最后調用Folder的getMessages()方法得到目錄中所有Message的數組。

    注意:對于POP3協議只有一個名為INBOX的Folder有效,而對于IMAP協議,我們可以訪問多個Folder(想想前面講的IMAP協議)。而且SUN在設計Folder的getMessages()方法時采取了很智能的方式:首先接收新郵件列表,然后再需要的時候(比如讀取郵件內容)才從郵件服務器讀取郵件內容。
    在讀取郵件時,我們可以用Message類的getContent()方法接收郵件或是writeTo()方法將郵件保存,getContent()方法只接收郵件內容(不包含郵件頭),而writeTo()方法將包括郵件頭。
    System.out.println(((MimeMessage)message).getContent());
    在讀取郵件內容后,別忘記了關閉Folder和Store。
    folder.close(aBoolean);store.close();
    傳遞給Folder.close()方法的boolean?類型參數表示是否在刪除操作郵件后更新Folder。?

    H.繼續向前進!
    在講解了以上的七個Java?Mail核心類定義和理解了簡單的代碼片斷后,下文將詳細講解怎樣使用這些類實現JavaMail?API所要完成的高級功能。

    五、使用JavaMail?API
    在明確了JavaMail?API的核心部分如何工作后,本人將帶領大家學習一些使用Java?Mail?API任務案例。
    1.發送郵件
    在獲得了Session后,建立并填入郵件信息,然后發送它到郵件服務器。這便是使用Java?Mail?API發送郵件的過程,在發送郵件之前,我們需要設置SMTP服務器:通過設置Properties的mail.smtp.host屬性。
    String?host?=?...;String?from?=?...;String?to?=?...;//?Get?system?propertiesProperties?props?=?System.getProperties();//?Setup?mail?serverprops.put("mail.smtp.host",?host);//?Get?sessionSession?session?=?Session.getDefaultInstance(props,?null);//?Define?messageMimeMessage?message?=?new?MimeMessage(session);message.setFrom(new?InternetAddress(from));message.addRecipient(Message.RecipientType.TO,???new?InternetAddress(to));message.setSubject("Hello?JavaMail");message.setText("Welcome?to?JavaMail");//?Send?messageTransport.send(message);
    由于建立郵件信息和發送郵件的過程中可能會拋出異常,所以我們需要將上面的代碼放入到try-catch結構塊中。

    2.接收郵件
    為了在讀取郵件,我們獲得了session,并且連接到了郵箱的相應store,打開相應的Folder,然后得到我們想要的郵件,當然別忘記了在結束時關閉連接。
    String?host?=?...;String?username?=?...;String?password?=?...;//?Create?empty?propertiesProperties?props?=?new?Properties();//?Get?sessionSession?session?=?Session.getDefaultInstance(props,?null);//?Get?the?storeStore?store?=?session.getStore("pop3");store.connect(host,?username,?password);//?Get?folderFolder?folder?=?store.getFolder("INBOX");folder.open(Folder.READ_ONLY);//?Get?directoryMessage?message[]?=?folder.getMessages();for?(int?i=0,?n=message.length;?i<n;?i++)?{???System.out.println(i?+?":?"?+?message[i].getFrom()[0]??????+?"\t"?+?message[i].getSubject());}//?Close?connection?folder.close(false);store.close();
    上面的代碼所作的是從郵箱中讀取每個郵件,并且顯示郵件的發信人地址和主題。從技術角度講,這里存在著一個異常的可能:當發信人地址為空時,getFrom()[0]將拋出異常。

    下面的代碼片斷有效的說明了如何讀取郵件內容,在顯示每個郵件發信人和主題后,將出現用戶提示從而得到用戶是否讀取該郵件的確認,如果輸入YES的話,我們可用Message.writeTo(java.io.OutputStream?os)方法將郵件內容輸出到控制臺上,關于Message.writeTo()的具體用法請看JavaMail?API。
    BufferedReader?reader?=?new?BufferedReader?(??new?InputStreamReader(System.in));//?Get?directoryMessage?message[]?=?folder.getMessages();for?(int?i=0,?n=message.length;?i<n;?i++)?{??System.out.println(i?+?":?"?+?message[i].getFrom()[0]?????+?"\t"?+?message[i].getSubject());??System.out.println("Do?you?want?to?read?message??"?+????"[YES?to?read/QUIT?to?end]");??String?line?=?reader.readLine();??if?("YES".equals(line))?{????message[i].writeTo(System.out);??}?else?if?("QUIT".equals(line))?{????break;??}}

    3.刪除郵件和標志
    設置與message相關的Flags是刪除郵件的常用方法。這些Flags表示了一些系統定義和用戶定義的不同狀態。在Flags類的內部類Flag中預定義了一些標志:
    Flags.Flag.ANSWERED
    Flags.Flag.DELETED
    Flags.Flag.DRAFT
    Flags.Flag.FLAGGED
    Flags.Flag.RECENT
    Flags.Flag.SEEN
    Flags.Flag.USER
    但需要在使用時注意的:標志存在并非意味著這個標志被所有的郵件服務器所支持。例如,對于刪除郵件的操作,POP協議不支持上面的任何一個。所以要確定哪些標志是被支持的??通過訪問一個已經打開的Folder對象的getPermanetFlags()方法,它將返回當前被支持的Flags類對象。
    刪除郵件時,我們可以設置郵件的DELETED標志:?
    message.setFlag(Flags.Flag.DELETED,?true);
    但是首先要采用READ_WRITE的方式打開Folder:
    folder.open(Folder.READ_WRITE);
    在對郵件進行刪除操作后關閉Folder時,需要傳遞一個true作為對刪除郵件的擦除確認。
    folder.close(true);
    Folder類中另一種用于刪除郵件的方法expunge()也同樣可刪除郵件,但是它并不為sun提供的POP3實現支持,而其它第三方提供的POP3實現支持或者并不支持這種方法。
    另外,介紹一種檢查某個標志是否被設置的方法:Message.isSet(Flags.Flag?flag)方法,其中參數為被檢查的標志。

    4.郵件認證
    我們在前面已經學會了如何使用Authenticator類來代替直接使用用戶名和密碼這兩字符串作為Session.getDefaultInstance()或者Session.getInstance()方法的參數。在前面的小試牛刀后,現在我們將了解到全面認識一下郵件認證。
    我們在此取代了直接使用郵件服務器主機名、用戶名、密碼這三個字符串作為連接到POP3?Store的方式,使用存儲了郵件服務器主機名信息的屬性文件,并在獲得Session時傳入自定義的Authenticator實例:
    //?Setup?propertiesProperties?props?=?System.getProperties();props.put("mail.pop3.host",?host);//?Setup?authentication,?get?sessionAuthenticator?auth?=?new?PopupAuthenticator();Session?session?=?Session.getDefaultInstance(props,?auth);//?Get?the?storeStore?store?=?session.getStore("pop3");store.connect();

    PopupAuthenticator類繼承了抽象類Authenticator,并且通過重載Authenticator類的getPasswordAuthentication()方法返回PasswordAuthentication類對象。而getPasswordAuthentication()方法的參數param是以逗號分割的用戶名、密碼組成的字符串。
    import?javax.mail.*;import?java.util.*;public?class?PopupAuthenticator?extends?Authenticator?{??public?PasswordAuthentication?getPasswordAuthentication(String?param)?{????String?username,?password;????StringTokenizer?st?=?new?StringTokenizer(param,?",");????username?=?st.nextToken();????password?=?st.nextToken();????return?new?PasswordAuthentication(username,?password);??}}

    5.回復郵件
    回復郵件的方法很簡單:使用Message類的reply()方法,通過配置回復郵件的收件人地址和主題(如果沒有提供主題的話,系統將默認將“Re:”作為郵件的主體),這里不需要設置任何的郵件內容,只要復制發信人或者reply-to到新的收件人。而reply()方法中的boolean參數表示是否將郵件回復給發送者(參數值為false),或是恢復給所有人(參數值為true)。
    補充一下,reply-to地址需要在發信時使用setReplyTo()方法設置。
    MimeMessage?reply?=?(MimeMessage)message.reply(false);reply.setFrom(new?InternetAddress("president@whitehouse.gov"));reply.setText("Thanks");Transport.send(reply);

    6.轉發郵件
    轉發郵件的過程不如前面的回復郵件那樣簡單,它將建立一個轉發郵件,這并非一個方法就能做到。
    每個郵件是由多個部分組成,每個部分稱為一個郵件體部分,是一個BodyPart類對象,對于MIME類型郵件來講就是MimeBodyPart類對象。這些郵件體包含在成為Multipart的容器中對于MIME類型郵件來講就是MimeMultiPart類對象。在轉發郵件時,我們建立一個文字郵件體部分和一個被轉發的文字郵件體部分,然后將這兩個郵件體放到一個Multipart中。說明一下,復制一個郵件內容到另一個郵件的方法是僅復制它的DataHandler(數據處理者)即可。這是由JavaBeans?Activation?Framework定義的一個類,它提供了對郵件內容的操作命令的訪問、管理了郵件內容操作,是不同的數據源和數據格式之間的一致性接口。
    //?Create?the?message?to?forwardMessage?forward?=?new?MimeMessage(session);//?Fill?in?headerforward.setSubject("Fwd:?"?+?message.getSubject());forward.setFrom(new?InternetAddress(from));forward.addRecipient(Message.RecipientType.TO,???new?InternetAddress(to));//?Create?your?new?message?partBodyPart?messageBodyPart?=?new?MimeBodyPart();messageBodyPart.setText(??"Here?you?go?with?the?original?message:\n\n");//?Create?a?multi-part?to?combine?the?partsMultipart?multipart?=?new?MimeMultipart();multipart.addBodyPart(messageBodyPart);//?Create?and?fill?part?for?the?forwarded?contentmessageBodyPart?=?new?MimeBodyPart();messageBodyPart.setDataHandler(message.getDataHandler());//?Add?part?to?multi?partmultipart.addBodyPart(messageBodyPart);//?Associate?multi-part?with?messageforward.setContent(multipart);//?Send?messageTransport.send(forward);

    7.使用附件
    附件作為與郵件相關的資源經常以文本、表格、圖片等格式出現,如流行的郵件客戶端一樣,我們可以用JavaMail?API從郵件中獲取附件或是發送帶有附件的郵件。

    A.發送帶有附件的郵件
    發送帶有附件的郵件的過程有些類似轉發郵件,我們需要建立一個完整郵件的各個郵件體部分,在第一個部分(即我們的郵件內容文字)后,增加一個具有DataHandler的附件而不是在轉發郵件時那樣復制第一個部分的DataHandler。

    如果我們將文件作為附件發送,那么要建立FileDataSource類型的對象作為附件數據源;如果從URL讀取數據作為附件發送,那么將要建立URLDataSource類型的對象作為附件數據源。

    然后將這個數據源(FileDataSource或是URLDataSource)對象作為DataHandler類構造方法的參數傳入,從而建立一個DataHandler對象作為數據源的DataHandler。

    接著將這個DataHandler設置為郵件體部分的DataHandler。這樣就完成了郵件體與附件之間的關聯工作,下面的工作就是BodyPart的setFileName()方法設置附件名為原文件名。

    最后將兩個郵件體放入到Multipart中,設置郵件內容為這個容器Multipart,發送郵件。
    //?Define?messageMessage?message?=?new?MimeMessage(session);message.setFrom(new?InternetAddress(from));message.addRecipient(Message.RecipientType.TO,???new?InternetAddress(to));message.setSubject("Hello?JavaMail?Attachment");//?Create?the?message?part?BodyPart?messageBodyPart?=?new?MimeBodyPart();//?Fill?the?messagemessageBodyPart.setText("Pardon?Ideas");Multipart?multipart?=?new?MimeMultipart();multipart.addBodyPart(messageBodyPart);//?Part?two?is?attachmentmessageBodyPart?=?new?MimeBodyPart();DataSource?source?=?new?FileDataSource(filename);messageBodyPart.setDataHandler(new?DataHandler(source));messageBodyPart.setFileName(filename);multipart.addBodyPart(messageBodyPart);//?Put?parts?in?messagemessage.setContent(multipart);//?Send?the?messageTransport.send(message);
    如果我們使用servlet實現發送帶有附件的郵件,則必須上傳附件給servlet,這時需要注意提交頁面form中對編碼類型的設置應為multipart/form-data。
    <FORM?ENCTYPE="multipart/form-data"?????method=post?action="/myservlet">???<INPUT?TYPE="file"?NAME="thefile">??<INPUT?TYPE="submit"?VALUE="Upload"></FORM>

    B.讀取郵件中的附件
    讀取郵件中的附件的過程要比發送它的過程復雜一點。因為帶有附件的郵件是多部分組成的,我們必須處理每一個部分獲得郵件的內容和附件。
    但是如何辨別郵件信息內容和附件呢?Sun在Part類(BodyPart類實現的接口類)中提供了getDisposition()方法讓開發者獲得郵件體部分的部署類型,當該部分是附件時,其返回之將是Part.ATTACHMENT。但附件也可以沒有部署類型的方式存在或者部署類型為Part.INLINE,無論部署類型為Part.ATTACHMENT還是Part.INLINE,我們都能把該郵件體部分導出保存。
    Multipart?mp?=?(Multipart)message.getContent();for?(int?i=0,?n=multipart.getCount();?i<n;?i++)?{??Part?part?=?multipart.getBodyPart(i));??String?disposition?=?part.getDisposition();??if?((disposition?!=?null)?&&???????((disposition.equals(Part.ATTACHMENT)?||????????(disposition.equals(Part.INLINE)))?{????saveFile(part.getFileName(),?part.getInputStream());??}}
    下列代碼中使用了saveFile方法是自定義的方法,它根據附件的文件名建立一個文件,如果本地磁盤上存在名為附件的文件,那么將在文件名后增加數字表示區別。然后從郵件體中讀取數據寫入到本地文件中(代碼省略)。
    //?from?saveFile()File?file?=?new?File(filename);for?(int?i=0;?file.exists();?i++)?{??file?=?new?File(filename+i);}
    以上是郵件體部分被正確設置的簡單例子,如果郵件體部分的部署類型為null,那么我們通過獲得郵件體部分的MIME類型來判斷其類型作相應的處理,代碼結構框架如下:
    if?(disposition?==?null)?{??//?Check?if?plain??MimeBodyPart?mbp?=?(MimeBodyPart)part;??if?(mbp.isMimeType("text/plain"))?{????//?Handle?plain??}?else?{????//?Special?non-attachment?cases?here?of?????//?image/gif,?text/html,?...??}...}

    8.處理HTML郵件
    前面的例子中發送的郵件都是以文本為內容的(除了附件),下面將介紹如何接收和發送基于HTML的郵件。
    A.發送HTML郵件
    假如我們需要發送一個HTML文件作為郵件內容,并使郵件客戶端在讀取郵件時獲取相關的圖片或者文字的話,只要設置郵件內容為html代碼,并設置內容類型為text/html即可:
    String?htmlText?=?"<H1>Hello</H1>"?+???"<img?src=\"http://www.jguru.com/images/logo.gif\">";message.setContent(htmlText,?"text/html"));
    請注意:這里的圖片并不是在郵件中內嵌的,而是在URL中定義的。郵件接收者只有在線時才能看到。
    在接收郵件時,如果我們使用JavaMail?API接收郵件的話是無法實現以HTML方式顯示郵件內容的。因為JavaMail?API郵件內容視為二進制流。所以要顯示HTML內容的郵件,我們必須使用JEditorPane或者第三方HTML展現組件。

    以下代碼顯示了如何使用JEditorPane顯示郵件內容:
    if?(message.getContentType().equals("text/html"))?{??String?content?=?(String)message.getContent();??JFrame?frame?=?new?JFrame();??JEditorPane?text?=?new?JEditorPane("text/html",?content);??text.setEditable(false);??JScrollPane?pane?=?new?JScrollPane(text);??frame.getContentPane().add(pane);??frame.setSize(300,?300);??frame.setDefaultCloseOperation(JFrame.DISPOSE_ON_CLOSE);??frame.show();}

    B.在郵件中包含圖片
    如果我們在郵件中使用HTML作為內容,那么最好將HTML中使用的圖片作為郵件的一部分,這樣無論是否在線都會正確的顯示HTML中的圖片。處理方法就是將HTML中用到的圖片作為郵件附件并使用特殊的cid?URL作為圖片的引用,這個cid就是對圖片附件的Content-ID頭的引用。
    處理內嵌圖片就像向郵件中添加附件一樣,不同之處在于我們必須通過設置圖片附件所在的郵件體部分的header中Content-ID為一個隨機字符串,并在HTML中img的src標記中設置為該字符串。這樣就完成了圖片附件與HTML的關聯。
    String?file?=?...;//?Create?the?messageMessage?message?=?new?MimeMessage(session);//?Fill?its?headersmessage.setSubject("Embedded?Image");message.setFrom(new?InternetAddress(from));message.addRecipient(Message.RecipientType.TO,???new?InternetAddress(to));//?Create?your?new?message?partBodyPart?messageBodyPart?=?new?MimeBodyPart();String?htmlText?=?"<H1>Hello</H1>"?+???"<img?src=\"cid:memememe\">";messageBodyPart.setContent(htmlText,?"text/html");//?Create?a?related?multi-part?to?combine?the?partsMimeMultipart?multipart?=?new?MimeMultipart("related");multipart.addBodyPart(messageBodyPart);//?Create?part?for?the?imagemessageBodyPart?=?new?MimeBodyPart();//?Fetch?the?image?and?associate?to?partDataSource?fds?=?new?FileDataSource(file);messageBodyPart.setDataHandler(new?DataHandler(fds));messageBodyPart.setHeader("Content-ID","<memememe>");//?Add?part?to?multi-partmultipart.addBodyPart(messageBodyPart);//?Associate?multi-part?with?messagemessage.setContent(multipart);

    9.在郵件中搜索短語
    JavaMail?API提供了過濾器機制,它被用來建立搜索短語。這個短語由javax.mail.search包中的SearchTerm抽象類來定義,在定義后我們便可以使用Folder的Search()方法在Folder中查找郵件:
    SearchTerm?st?=?...;Message[]?msgs?=?folder.search(st);
    下面有22個不同的類(繼承了SearchTerm類)供我們使用:
    AND?terms?(class?AndTerm)
    OR?terms?(class?OrTerm)
    NOT?terms?(class?NotTerm)
    SENT?DATE?terms?(class?SentDateTerm)
    CONTENT?terms?(class?BodyTerm)
    HEADER?terms?(FromTerm?/?FromStringTerm,?RecipientTerm?/?RecipientStringTerm,?SubjectTerm,?etc.)
    使用這些類定義的斷語集合,我們可以構造一個邏輯表達式,并在Folder中進行搜索。下面是一個實例:在Folder中搜索郵件主題含有“ADV”字符串或者發信人地址為friend@public.com的郵件。
    SearchTerm?st?=???new?OrTerm(????new?SubjectTerm("ADV:"),?????new?FromStringTerm("friend@public.com"));Message[]?msgs?=?folder.search(st);

    六、參考資源
    JavaMail?API?Home
    Sun’s?JavaMail?API基礎
    JavaBeans?Activation?Framework?Home
    javamail-interest?mailing?list
    Sun's?JavaMail?FAQ
    jGuru's?JavaMail?FAQ
    Third?Party?Products?List
    七、代碼下載
    http://java.sun.com/developer/onlineTraining/JavaMail/exercises.html?

    posted @ 2006-06-09 10:19 nbt 閱讀(250) | 評論 (0)編輯 收藏

    Hibernate---通過XDoclet(ant)生成Hibernate映射文件(轉載)

    通過XDoclet可以我們的精力放在編寫java源文件上。

    具體來說就是:
    只有Java: java--->XDoclet(hibernatedoclet)--->Hbm---->SchemaExport(schemaexport,hbm2ddl)---->數據表

    1:java源文件編寫

    /*
    ?*?Created?on?2006-4-7
    ?
    */


    package ?com.entity;

    /**
    ?*?
    @author ?jkallen
    ?*?@hibernate.class?lazy="true"?table="syn_dept"
    ?*?@hibernate.cache?usage="read-write"
    ?
    */

    public ? class ?SynDepartment? {
    ?
    ?
    /** ?主鍵?id */
    ?
    private ?Long?id;
    ?
    /** ?部門名稱 */
    ?
    private ?String?code_name;
    ?
    ?
    /**
    ??*?
    @return ?Returns?the?id.
    ??*?@hibernate.id?generator-class="native"?column="id"
    ??
    */

    ????
    public ?Long?getId()? {
    ??
    return ?id;
    ?}

    ?
    public ? void ?setId(Long?id)? {
    ??
    this .id? = ?id;
    ?}

    ?
    /**
    ????*?
    @return ?Returns?the?code_name.
    ????*?@hibernate.property?column?=?"code_name"
    ????
    */

    ?
    public ?String?getCode_name()? {
    ??
    return ?code_name;
    ?}

    ?
    public ? void ?setCode_name(String?code_name)? {
    ??
    this .code_name? = ?code_name;
    ?}

    }




    這里用到了幾種@hibernate標記的用法
    @hibernate.class標記指定類的映射代碼,lazy="true" table="syn_dept"則如
    hibernate的映射文件class元素的屬性值具有相同的意義
    @hibernate.id標記指定類的OID映射代碼
    @hibernate.property標記指定類的屬性映射代碼
    另外還可能用到@hibernate.set(如一對多的情況下)

    2:XDoclet--->Hbm(寫在build.xml文件中,ANT運行)

    < target? name ="toHbm" ?
    ??depends
    ="compileEntity" ?
    ??description
    ="Generate?hibernate?mapping?documents" >
    ??
    < hibernatedoclet? destdir ="${generated.dir}" >
    ???
    < fileset? dir ="${src.dir}" >
    ????
    < include? name ="**/entity/*.java" ? />
    ???
    </ fileset >
    ???
    < hibernate? version ="2.0" ? />
    ??
    </ hibernatedoclet >

    ??
    < copy? todir ="${classes.dir}" >
    ???
    < fileset? dir ="${generated.dir}" ? />
    ??
    </ copy >
    ?
    </ target >


    通過hibernatedoclet就可以生成SynDepartment.hbm.xml映射文件
    fileset顧名思義就是過濾文件了。
    注:compileEntity--編譯java源文件(自定義)

    3:SchemaExport---->數據表

    < target? name ="toddl" ?depends ="init" >
    ??
    < schemaexport? properties ="${classes.dir}/hibernate.properties" ?
    ???quiet
    ="no" ?text ="no" ?drop ="no" ????
    ???delimiter
    ="&#xd;&#xa;go&#xd;&#xa;" ?output ="${sql.dir}/${synup.sql.file}"
    ???
    >
    ???
    < fileset? refid ="hibernate.synup.mapping.files" ? />
    ??
    </ schemaexport >
    ??
    < echo? message ="Output?sql?to?file:?${sql.dir}/${sql.file}" ? />
    ?
    </ target >
    ?
    < fileset? id ="hibernate.synup.mapping.files" ?dir ="${classes.dir}" >
    ??
    < include? name ="**/entity/*.hbm.xml" ? />
    ?
    </ fileset >


    ?通過schemaexport就向DB中生成table了。其中可能用到如下的一些屬性:
    ?quiet:如果為yes,表示不把子DDL腳本輸出到控制臺
    ?drop:如果為yes,只執行刪除數據庫中的操作,但不創建新的表
    ?text:如果為yes,只會生成DDL腳本文件,但不會在數據庫中執行DDL腳本
    ?output:指定存放DDL腳本文件的目錄
    ?config:設定基于XML格式的配置文件, hbm2ddl(schemaexport)工具從這個文件中讀取數據庫的配置信息
    ?properties:設定基于java屬性文件格式的配置文件,hbm2ddl(schemaexport)工具從這個文件中讀取DB的配置信息
    ?format:設定DDL腳本中SQL語句的格式
    ?delimiter:為DDL腳本設置行結束符
    ?
    ?在ANT中執行:
    ?<target name="initOnlySynup" depends="toHbm,toddl">
    ?</target>
    ?
    ?OK,最后生成的映射文件如下:

    <? xml?version="1.0" ?>

    <! DOCTYPE?hibernate-mapping?PUBLIC
    ????"-//Hibernate/Hibernate?Mapping?DTD?2.0//EN"?
    ????"http://hibernate.sourceforge.net/hibernate-mapping-2.0.dtd"
    >

    < hibernate-mapping >
    ????
    < class
    ????????
    name ="com.SynDepartment"
    ????????table
    ="syn_dept"
    ????????dynamic-update
    ="false"
    ????????dynamic-insert
    ="false"
    ????
    >
    ????????
    < cache? usage ="read-write" ? />

    ????????
    < id
    ????????????
    name ="id"
    ????????????column
    ="id"
    ????????????type
    ="java.lang.Long"
    ????????
    >
    ????????????
    < generator? class ="native" >
    ????????????
    </ generator >
    ????????
    </ id >

    ????????
    < property
    ????????????
    name ="code_name"
    ????????????type
    ="java.lang.String"
    ????????????update
    ="true"
    ????????????insert
    ="true"
    ????????????access
    ="property"
    ????????????column
    ="code_name"
    ????????
    />

    ????????
    <!--
    ????????????To?add?non?XDoclet?property?mappings,?create?a?file?named
    ????????????????hibernate-properties-SynDepartment.xml
    ????????????containing?the?additional?properties?and?place?it?in?your?merge?dir.
    ????????
    -->

    ????
    </ class >

    </ hibernate-mapping >


    ?控制臺中部分信息如下:

    [schemaexport]?drop?table?syn_dept?cascade?constraints
    [schemaexport]?go
    [schemaexport]?drop?sequence?hibernate_sequence
    [schemaexport]?go
    [schemaexport]?create?table?syn_dept?(
    [schemaexport]?id?number(19,0)?not?null,
    [schemaexport]?code_name?varchar2(255),
    [schemaexport]?primary?key?(id)
    [schemaexport]?)

    DB中已經生成syn_dept表了,快去看下吧!

    關于Xdoclet 中的hibernate標簽更多信息可以參考:
    http://xdoclet.sourceforge.net/xdoclet/tags/hibernate-tags.html#@hibernate_collection-key__0__1_
    我還在一個網友的博客上看到了他對此的漢化:
    http://blog.csdn.net/fasttalk/archive/2005/09/19/484615.aspx

    posted @ 2006-06-09 09:24 nbt 閱讀(493) | 評論 (0)編輯 收藏

    精美SQL語句(轉載)

    asc 按升序排列
    desc 按降序排列

    下列語句部分是Mssql語句,不可以在access中使用。

    SQL分類:
    DDL—數據定義語言(CREATE,ALTER,DROP,DECLARE)
    DML—數據操縱語言(SELECT,DELETE,UPDATE,INSERT)
    DCL—數據控制語言(GRANT,REVOKE,COMMIT,ROLLBACK)

    首先,簡要介紹基礎語句:
    1、說明:創建數據庫
    CREATE DATABASE database-name
    2、說明:刪除數據庫
    drop database dbname
    3、說明:備份sql server
    --- 創建 備份數據的 device
    USE master
    EXEC sp_addumpdevice 'disk', 'testBack', 'c:\mssql7backup\MyNwind_1.dat'
    --- 開始 備份
    BACKUP DATABASE pubs TO testBack
    4、說明:創建新表
    create table tabname(col1 type1 [not null] [primary key],col2 type2 [not null],..)
    根據已有的表創建新表:
    A:create table tab_new like tab_old (使用舊表創建新表)
    B:create table tab_new as select col1,col2… from tab_old definition only
    5、說明:刪除新表drop table tabname
    6、說明:增加一個列
    Alter table tabname add column col type
    注:列增加后將不能刪除。DB2中列加上后數據類型也不能改變,唯一能改變的是增加varchar類型的長度。
    7、說明:添加主鍵: Alter table tabname add primary key(col)
    說明:刪除主鍵: Alter table tabname drop primary key(col)
    8、說明:創建索引:create [unique] index idxname on tabname(col….)
    刪除索引:drop index idxname
    注:索引是不可更改的,想更改必須刪除重新建。
    9、說明:創建視圖:create view viewname as select statement
    刪除視圖:drop view viewname
    10、說明:幾個簡單的基本的sql語句
    選擇:select * from table1 where 范圍
    插入:insert into table1(field1,field2) values(value1,value2)
    刪除:delete from table1 where 范圍
    更新:update table1 set field1=value1 where 范圍
    查找:select * from table1 where field1 like ’%value1%’ ---like的語法很精妙,查資料!
    排序:select * from table1 order by field1,field2 [desc]
    總數:select count as totalcount from table1
    求和:select sum(field1) as sumvalue from table1
    平均:select avg(field1) as avgvalue from table1
    最大:select max(field1) as maxvalue from table1
    最小:select min(field1) as minvalue from table1
    11、說明:幾個高級查詢運算詞
    A: UNION 運算符
    UNION 運算符通過組合其他兩個結果表(例如 TABLE1 和 TABLE2)并消去表中任何重復行而派生出一個結果表。當 ALL 隨 UNION 一起使用時(即 UNION ALL),不消除重復行。兩種情況下,派生表的每一行不是來自 TABLE1 就是來自 TABLE2。
    B: EXCEPT 運算符
    EXCEPT 運算符通過包括所有在 TABLE1 中但不在 TABLE2 中的行并消除所有重復行而派生出一個結果表。當 ALL 隨 EXCEPT 一起使用時 (EXCEPT ALL),不消除重復行。
    C: INTERSECT 運算符
    INTERSECT 運算符通過只包括 TABLE1 和 TABLE2 中都有的行并消除所有重復行而派生出一個結果表。當 ALL 隨 INTERSECT 一起使用時 (INTERSECT ALL),不消除重復行。
    注:使用運算詞的幾個查詢結果行必須是一致的。
    12、說明:使用外連接
    A、left outer join:
    左外連接(左連接):結果集幾包括連接表的匹配行,也包括左連接表的所有行。
    SQL: select a.a, a.b, a.c, b.c, b.d, b.f from a LEFT OUT JOIN b ON a.a = b.c
    B:right outer join:
    右外連接(右連接):結果集既包括連接表的匹配連接行,也包括右連接表的所有行。
    C:full outer join:
    全外連接:不僅包括符號連接表的匹配行,還包括兩個連接表中的所有記錄。

    其次,大家來看一些不錯的sql語句
    1、說明:復制表(只復制結構,源表名:a 新表名:b) (Access可用)
    法一:select * into b from a where 1<>1
    法二:select top 0 * into b from a

    2、說明:拷貝表(拷貝數據,源表名:a 目標表名:b) (Access可用)
    insert into b(a, b, c) select d,e,f from b;

    3、說明:跨數據庫之間表的拷貝(具體數據使用絕對路徑) (Access可用)
    insert into b(a, b, c) select d,e,f from b in ‘具體數據庫’ where 條件
    例子:..from b in '"&Server.MapPath(".")&"\data.mdb" &"' where..

    4、說明:子查詢(表名1:a 表名2:b)
    select a,b,c from a where a IN (select d from b ) 或者: select a,b,c from a where a IN (1,2,3)

    5、說明:顯示文章、提交人和最后回復時間
    select a.title,a.username,b.adddate from table a,(select max(adddate) adddate from table where table.title=a.title) b

    6、說明:外連接查詢(表名1:a 表名2:b)
    select a.a, a.b, a.c, b.c, b.d, b.f from a LEFT OUT JOIN b ON a.a = b.c

    7、說明:在線視圖查詢(表名1:a )
    select * from (SELECT a,b,c FROM a) T where t.a > 1;

    8、說明:between的用法,between限制查詢數據范圍時包括了邊界值,not between不包括
    select * from table1 where time between time1 and time2
    select a,b,c, from table1 where a not between 數值1 and 數值2

    9、說明:in 的使用方法
    select * from table1 where a [not] in (‘值1’,’值2’,’值4’,’值6’)

    10、說明:兩張關聯表,刪除主表中已經在副表中沒有的信息
    delete from table1 where not exists ( select * from table2 where table1.field1=table2.field1 )

    11、說明:四表聯查問題:
    select * from a left inner join b on a.a=b.b right inner join c on a.a=c.c inner join d on a.a=d.d where .....

    12、說明:日程安排提前五分鐘提醒
    SQL: select * from 日程安排 where datediff('minute',f開始時間,getdate())>5

    13、說明:一條sql 語句搞定數據庫分頁
    select top 10 b.* from (select top 20 主鍵字段,排序字段 from 表名 order by 排序字段 desc) a,表名 b where b.主鍵字段 = a.主鍵字段 order by a.排序字段

    14、說明:前10條記錄
    select top 10 * form table1 where 范圍

    15、說明:選擇在每一組b值相同的數據中對應的a最大的記錄的所有信息(類似這樣的用法可以用于論壇每月排行榜,每月熱銷產品分析,按科目成績排名,等等.)
    select a,b,c from tablename ta where a=(select max(a) from tablename tb where tb.b=ta.b)

    16、說明:包括所有在 TableA 中但不在 TableB和TableC 中的行并消除所有重復行而派生出一個結果表
    (select a from tableA ) except (select a from tableB) except (select a from tableC)

    17、說明:隨機取出10條數據
    select top 10 * from tablename order by newid()

    18、說明:隨機選擇記錄
    select newid()

    19、說明:刪除重復記錄
    Delete from tablename where id not in (select max(id) from tablename group by col1,col2,...)

    20、說明:列出數據庫里所有的表名
    select name from sysobjects where type='U'

    21、說明:列出表里的所有的
    select name from syscolumns where id=object_id('TableName')

    22、說明:列示type、vender、pcs字段,以type字段排列,case可以方便地實現多重選擇,類似select 中的case。
    select type,sum(case vender when 'A' then pcs else 0 end),sum(case vender when 'C' then pcs else 0 end),sum(case vender when 'B' then pcs else 0 end) FROM tablename group by type
    顯示結果:
    type vender pcs
    電腦 A 1
    電腦 A 1
    光盤 B 2
    光盤 A 2
    手機 B 3
    手機 C 3

    23、說明:初始化表table1

    TRUNCATE TABLE table1

    24、說明:選擇從10到15的記錄
    select top 5 * from (select top 15 * from table order by id asc) table_別名 order by id desc
      
    隨機選擇數據庫記錄的方法(使用Randomize函數,通過SQL語句實現)
      對存儲在數據庫中的數據來說,隨機數特性能給出上面的效果,但它們可能太慢了些。你不能要求ASP“找個隨機數”然后打印出來。實際上常見的解決方案是建立如下所示的循環:
    Randomize
    RNumber = Int(Rnd*499) +1
     
    While Not objRec.EOF
    If objRec("ID") = RNumber THEN
    ... 這里是執行腳本 ...
    end if
    objRec.MoveNext
    Wend
     
      這很容易理解。首先,你取出1到500范圍之內的一個隨機數(假設500就是數據庫內記錄的總數)。然后,你遍歷每一記錄來測試ID 的值、檢查其是否匹配RNumber。滿足條件的話就執行由THEN 關鍵字開始的那一塊代碼。假如你的RNumber 等于495,那么要循環一遍數據庫花的時間可就長了。雖然500這個數字看起來大了些,但相比更為穩固的企業解決方案這還是個小型數據庫了,后者通常在一個數據庫內就包含了成千上萬條記錄。這時候不就死定了?
      采用SQL,你就可以很快地找出準確的記錄并且打開一個只包含該記錄的recordset,如下所示:
    Randomize
    RNumber = Int(Rnd*499) + 1
     
    SQL = "SELECT * FROM Customers WHERE ID = " & RNumber
     
    set objRec = ObjConn.Execute(SQL)
    Response.WriteRNumber & " = " & objRec("ID") & " " & objRec("c_email")
     
      不必寫出RNumber 和ID,你只需要檢查匹配情況即可。只要你對以上代碼的工作滿意,你自可按需操作“隨機”記錄。Recordset沒有包含其他內容,因此你很快就能找到你需要的記錄這樣就大大降低了處理時間。
    再談隨機數
      現在你下定決心要榨干Random 函數的最后一滴油,那么你可能會一次取出多條隨機記錄或者想采用一定隨機范圍內的記錄。把上面的標準Random 示例擴展一下就可以用SQL應對上面兩種情況了。
      為了取出幾條隨機選擇的記錄并存放在同一recordset內,你可以存儲三個隨機數,然后查詢數據庫獲得匹配這些數字的記錄:
    SQL = "SELECT * FROM Customers WHERE ID = " & RNumber & " OR ID = " & RNumber2 & " OR ID = " & RNumber3
     
      假如你想選出10條記錄(也許是每次頁面裝載時的10條鏈接的列表),你可以用BETWEEN 或者數學等式選出第一條記錄和適當數量的遞增記錄。這一操作可以通過好幾種方式來完成,但是 SELECT 語句只顯示一種可能(這里的ID 是自動生成的號碼):
    SQL = "SELECT * FROM Customers WHERE ID BETWEEN " & RNumber & " AND " & RNumber & "+ 9"

      注意:以上代碼的執行目的不是檢查數據庫內是否有9條并發記錄。

     
    隨機讀取若干條記錄,測試過
    Access語法:SELECT top 10 * From 表名 ORDER BY Rnd(id)
    Sql server:select top n * from 表名 order by newid()
    mysqlelect * From 表名 Order By rand() Limit n
    Access左連接語法(最近開發要用左連接,Access幫助什么都沒有,網上沒有Access的SQL說明,只有自己測試, 現在記下以備后查)
    語法elect table1.fd1,table1,fd2,table2.fd2 From table1 left join table2 on table1.fd1,table2.fd1 where ...
    使用SQL語句 用...代替過長的字符串顯示
    語法:
    SQL數據庫:select case when len(field)>10 then left(field,10)+'...' else field end as news_name,news_id from tablename
    Access數據庫:SELECT iif(len(field)>2,left(field,2)+'...',field) FROM tablename;
     
    Conn.Execute說明
    Execute方法
      該方法用于執行SQL語句。根據SQL語句執行后是否返回記錄集,該方法的使用格式分為以下兩種:

     1.執行SQL查詢語句時,將返回查詢得到的記錄集。用法為:
        Set 對象變量名=連接對象.Execute("SQL 查詢語言")
       Execute方法調用后,會自動創建記錄集對象,并將查詢結果存儲在該記錄對象中,通過Set方法,將記錄集賦給指定的對象保存,以后對象變量就代表了該記錄集對象。

        2.執行SQL的操作性語言時,沒有記錄集的返回。此時用法為:
        連接對象.Execute "SQL 操作性語句" [, RecordAffected][, Option]
          ·RecordAffected 為可選項,此出可放置一個變量,SQL語句執行后,所生效的記錄數會自動保存到該變量中。通過訪問該變量,就可知道SQL語句隊多少條記錄進行了操作。
          ·Option 可選項,該參數的取值通常為adCMDText,它用于告訴ADO,應該將Execute方法之后的第一個字符解釋為命令文本。通過指定該參數,可使執行更高效。

    ·BeginTrans、RollbackTrans、CommitTrans方法
      這三個方法是連接對象提供的用于事務處理的方法。BeginTrans用于開始一個事物;RollbackTrans用于回滾事務;CommitTrans用于提交所有的事務處理結果,即確認事務的處理。
      事務處理可以將一組操作視為一個整體,只有全部語句都成功執行后,事務處理才算成功;若其中有一個語句執行失敗,則整個處理就算失敗,并恢復到處里前的狀態。
      BeginTrans和CommitTrans用于標記事務的開始和結束,在這兩個之間的語句,就是作為事務處理的語句。判斷事務處理是否成功,可通過連接對象的Error集合來實現,若Error集合的成員個數不為0,則說明有錯誤發生,事務處理失敗。Error集合中的每一個Error對象,代表一個錯誤信息。

    posted @ 2006-06-05 12:08 nbt 閱讀(577) | 評論 (0)編輯 收藏

    Java Reflection (JAVA反射) ----轉載


    ??? Reflection 是 Java 程序開發語言的特征之一,它允許運行中的 Java 程序對自身進行檢查,或者說“自審”,并能直接操作程序的內部屬性。例如,使用它能獲得 Java 類中各成員的名稱并顯示出來。Java 的這一能力在實際應用中也許用得不是很多,但是在其它的程序設計語言中根本就不存在這一特性。例如,Pascal、C 或者 C++ 中就沒有辦法在程序中獲得函數定義相關的信息。
    JavaBean 是 reflection 的實際應用之一,它能讓一些工具可視化的操作軟件組件。這些工具通過 reflection 動態的載入并取得 Java 組件(類) 的屬性。

    1. 一個簡單的例子
    考慮下面這個簡單的例子,讓我們看看 reflection 是如何工作的。
    import java.lang.reflect.*;
    public class DumpMethods {
    ??? public static void main(String args[]) {
    ??????? try {
    ??????????? Class c = Class.forName(args[0]);
    ??????????? Method m[] = c.getDeclaredMethods();
    ??????????? for (int i = 0; i < m.length; i++)
    ??????????????? System.out.println(m[i].toString());
    ??????? } catch (Throwable e) {
    ??????????? System.err.println(e);
    ??????? }
    ??? }
    }


    按如下語句執行:
    java DumpMethods java.util.Stack

    它的結果輸出為:
    public java.lang.Object java.util.Stack.push(java.lang.Object)
    public synchronized java.lang.Object java.util.Stack.pop()
    public synchronized java.lang.Object java.util.Stack.peek()
    public boolean java.util.Stack.empty()
    public synchronized int java.util.Stack.search(java.lang.Object)

    這樣就列出了java.util.Stack 類的各方法名以及它們的限制符和返回類型。

    這個程序使用 Class.forName 載入指定的類,然后調用 getDeclaredMethods 來獲取這個類中定義了的方法列表。java.lang.reflect.Methods 是用來描述某個類中單個方法的一個類。


    2.開始使用 Reflection
    用于 reflection 的類,如 Method,可以在 java.lang.relfect 包中找到。使用這些類的時候必須要遵循三個步驟:第一步是獲得你想操作的類的 java.lang.Class 對象。在運行中的 Java 程序中,用 java.lang.Class 類來描述類和接口等。

    下面就是獲得一個 Class 對象的方法之一:
    Class c = Class.forName("java.lang.String");
    這條語句得到一個 String 類的類對象。還有另一種方法,如下面的語句:
    Class c = int.class;?? 或者?? Class c = Integer.TYPE;

    它們可獲得基本類型的類信息。其中后一種方法中訪問的是基本類型的封裝類 (如 Integer) 中預先定義好的 TYPE 字段。

    第二步是調用諸如 getDeclaredMethods 的方法,以取得該類中定義的所有方法的列表。

    一旦取得這個信息,就可以進行第三步了——使用 reflection API 來操作這些信息,如下面這段代碼:
    Class c = Class.forName("java.lang.String");
    Method m[] = c.getDeclaredMethods();
    System.out.println(m[0].toString());
    它將以文本方式打印出 String 中定義的第一個方法的原型。
    在下面的例子中,這三個步驟將為使用 reflection 處理特殊應用程序提供例證。



    模擬 instanceof 操作符
    得到類信息之后,通常下一個步驟就是解決關于 Class 對象的一些基本的問題。例如,Class.isInstance 方法可以用于模擬 instanceof 操作符:
    class A {
    }

    public class instance1 {
    ??? public static void main(String args[]) {
    ??????? try {
    ??????????? Class cls = Class.forName("A");
    ??????????? boolean b1 = cls.isInstance(new Integer(37));
    ??????????? System.out.println(b1);
    ??????????? boolean b2 = cls.isInstance(new A());
    ??????????? System.out.println(b2);
    ??????? } catch (Throwable e) {
    ??????????? System.err.println(e);
    ??????? }
    ??? }
    }

    在這個例子中創建了一個 A 類的 Class 對象,然后檢查一些對象是否是 A 的實例。Integer(37) 不是,但 new A() 是。


    3.找出類的方法
    找出一個類中定義了些什么方法,這是一個非常有價值也非常基礎的 reflection 用法。下面的代碼就實現了這一用法:
    import java.lang.reflect.*;
    public class method1 {
    ??? private int f1(Object p, int x) throws NullPointerException {
    ??????? if (p == null)
    ??????????? throw new NullPointerException();
    ??????? return x;
    ??? }
    ??? public static void main(String args[]) {
    ??????? try {
    ??????????? Class cls = Class.forName("method1");
    ??????????? Method methlist[] = cls.getDeclaredMethods();
    ??????????? for (int i = 0; i < methlist.length; i++) {
    ??????????????? Method m = methlist[i];
    ??????????????? System.out.println("name = " + m.getName());
    ??????????????? System.out.println("decl class = " + m.getDeclaringClass());
    ??????????????? Class pvec[] = m.getParameterTypes();
    ??????????????? for (int j = 0; j < pvec.length; j++)
    ??????????????????? System.out.println("param #" + j + " " + pvec[j]);
    ??????????????? Class evec[] = m.getExceptionTypes();
    ??????????????? for (int j = 0; j < evec.length; j++)
    ??????????????????? System.out.println("exc #" + j + " " + evec[j]);
    ??????????????? System.out.println("return type = " + m.getReturnType());
    ??????????????? System.out.println("-----");
    ??????????? }
    ??????? } catch (Throwable e) {
    ??????????? System.err.println(e);
    ??????? }
    ??? }
    }


    這個程序首先取得 method1 類的描述,然后調用 getDeclaredMethods 來獲取一系列的 Method 對象,它們分別描述了定義在類中的每一個方法,包括 public 方法、protected 方法、package 方法和 private 方法等。如果你在程序中使用 getMethods 來代替 getDeclaredMethods,你還能獲得繼承來的各個方法的信息。

    取得了 Method 對象列表之后,要顯示這些方法的參數類型、異常類型和返回值類型等就不難了。這些類型是基本類型還是類類型,都可以由描述類的對象按順序給出。

    輸出的結果如下:
    name = f1
    decl class = class method1
    param #0 class java.lang.Object
    param #1 int
    exc #0 class java.lang.NullPointerException
    return type = int
    -----
    name = main
    decl class = class method1
    param #0 class [Ljava.lang.String;
    return type = void
    -----


    4.獲取構造器信息
    獲取類構造器的用法與上述獲取方法的用法類似,如:
    import java.lang.reflect.*;
    public class constructor1 {
    ??? public constructor1() {
    ??? }

    ??? protected constructor1(int i, double d) {
    ??? }

    ??? public static void main(String args[]) {
    ??????? try {
    ??????????? Class cls = Class.forName("constructor1");
    ??????????? Constructor ctorlist[] = cls.getDeclaredConstructors();
    ??????????? for (int i = 0; i < ctorlist.length; i++) {
    ??????????????? Constructor ct = ctorlist[i];
    ??????????????? System.out.println("name = " + ct.getName());
    ??????????????? System.out.println("decl class = " + ct.getDeclaringClass());
    ??????????????? Class pvec[] = ct.getParameterTypes();
    ??????????????? for (int j = 0; j < pvec.length; j++)
    ??????????????????? System.out.println("param #" + j + " " + pvec[j]);
    ??????????????? Class evec[] = ct.getExceptionTypes();
    ??????????????? for (int j = 0; j < evec.length; j++)
    ??????????????????? System.out.println("exc #" + j + " " + evec[j]);
    ??????????????? System.out.println("-----");
    ??????????? }
    ??????? } catch (Throwable e) {
    ??????????? System.err.println(e);
    ??????? }
    ??? }
    }


    這個例子中沒能獲得返回類型的相關信息,那是因為構造器沒有返回類型。
    這個程序運行的結果是:
    name = constructor1
    decl class = class constructor1
    -----
    name = constructor1
    decl class = class constructor1
    param #0 int
    param #1 double
    -----

    5.獲取類的字段(域)
    找出一個類中定義了哪些數據字段也是可能的,下面的代碼就在干這個事情:
    import java.lang.reflect.*;

    public class field1 {
    ??? private double d;
    ??? public static final int i = 37;
    ??? String s = "testing";

    ?????public static void main(String args[]) {
    ??????? try {
    ??????????? Class cls = Class.forName("field1");
    ??????????? Field fieldlist[] = cls.getDeclaredFields();
    ??????????? for (int i = 0; i < fieldlist.length; i++) {
    ??????????????? Field fld = fieldlist[i];
    ??????????????? System.out.println("name = " + fld.getName());
    ??????????????? System.out.println("decl class = " + fld.getDeclaringClass());
    ??????????????? System.out.println("type = " + fld.getType());
    ??????????????? int mod = fld.getModifiers();
    ??????????????? System.out.println("modifiers = " + Modifier.toString(mod));
    ??????????????? System.out.println("-----");
    ??????????? }
    ??????? } catch (Throwable e) {
    ??????????? System.err.println(e);
    ??????? }
    ??? }
    }


    這個例子和前面那個例子非常相似。例中使用了一個新東西 Modifier,它也是一個 reflection 類,用來描述字段成員的修飾語,如“private int”。這些修飾語自身由整數描述,而且使用 Modifier.toString 來返回以“官方”順序排列的字符串描述 (如“static”在“final”之前)。這個程序的輸出是:
    name = d
    decl class = class field1
    type = double
    modifiers = private
    -----
    name = i
    decl class = class field1
    type = int
    modifiers = public static final
    -----
    name = s
    decl class = class field1
    type = class java.lang.String
    modifiers =
    -----

    和獲取方法的情況一下,獲取字段的時候也可以只取得在當前類中申明了的字段信息 (getDeclaredFields),或者也可以取得父類中定義的字段 (getFields) 。


    6.根據方法的名稱來執行方法
    文本到這里,所舉的例子無一例外都與如何獲取類的信息有關。我們也可以用 reflection 來做一些其它的事情,比如執行一個指定了名稱的方法。下面的示例演示了這一操作:
    import java.lang.reflect.*;
    public class method2 {
    ??? public int add(int a, int b) {
    ??????? return a + b;
    ??? }
    ??? public static void main(String args[]) {
    ??????? try {
    ??????????? Class cls = Class.forName("method2");
    ??????????? Class partypes[] = new Class[2];
    ??????????? partypes[0] = Integer.TYPE;
    ??????????? partypes[1] = Integer.TYPE;
    ??????????? Method meth = cls.getMethod("add", partypes);
    ??????????? method2 methobj = new method2();
    ??????????? Object arglist[] = new Object[2];
    ??????????? arglist[0] = new Integer(37);
    ??????????? arglist[1] = new Integer(47);
    ??????????? Object retobj = meth.invoke(methobj, arglist);
    ??????????? Integer retval = (Integer) retobj;
    ??????????? System.out.println(retval.intValue());
    ??????? } catch (Throwable e) {
    ??????????? System.err.println(e);
    ??????? }
    ??? }
    }


    假如一個程序在執行的某處的時候才知道需要執行某個方法,這個方法的名稱是在程序的運行過程中指定的 (例如,JavaBean 開發環境中就會做這樣的事),那么上面的程序演示了如何做到。


    上例中,getMethod 用于查找一個具有兩個整型參數且名為 add 的方法。找到該方法并創建了相應的 Method 對象之后,在正確的對象實例中執行它。執行該方法的時候,需要提供一個參數列表,這在上例中是分別包裝了整數 37 和 47 的兩個 Integer 對象。執行方法的返回的同樣是一個 Integer 對象,它封裝了返回值 84。


    7.創建新的對象
    對于構造器,則不能像執行方法那樣進行,因為執行一個構造器就意味著創建了一個新的對象 (準確的說,創建一個對象的過程包括分配內存和構造對象)。所以,與上例最相似的例子如下:
    import java.lang.reflect.*;
    public class constructor2 {
    ??? public constructor2() {
    ??? }

    ??? public constructor2(int a, int b) {
    ??????? System.out.println("a = " + a + " b = " + b);
    ??? }

    ??? public static void main(String args[]) {
    ??????? try {
    ??????????? Class cls = Class.forName("constructor2");
    ??????????? Class partypes[] = new Class[2];
    ??????????? partypes[0] = Integer.TYPE;
    ??????????? partypes[1] = Integer.TYPE;
    ??????????? Constructor ct = cls.getConstructor(partypes);
    ??????????? Object arglist[] = new Object[2];
    ??????????? arglist[0] = new Integer(37);
    ??????????? arglist[1] = new Integer(47);
    ??????????? Object retobj = ct.newInstance(arglist);
    ??????? } catch (Throwable e) {
    ??????????? System.err.println(e);
    ??????? }
    ??? }
    }

    根據指定的參數類型找到相應的構造函數并執行它,以創建一個新的對象實例。使用這種方法可以在程序運行時動態地創建對象,而不是在編譯的時候創建對象,這一點非常有價值。


    8.改變字段(域)的值
    reflection 的還有一個用處就是改變對象數據字段的值。reflection 可以從正在運行的程序中根據名稱找到對象的字段并改變它,下面的例子可以說明這一點:
    import java.lang.reflect.*;
    public class field2 {
    ??? public double d;
    ??? public static void main(String args[]) {
    ??????? try {
    ??????????? Class cls = Class.forName("field2");
    ??????????? Field fld = cls.getField("d");
    ??????????? field2 f2obj = new field2();
    ??????????? System.out.println("d = " + f2obj.d);
    ??????????? fld.setDouble(f2obj, 12.34);
    ??????????? System.out.println("d = " + f2obj.d);
    ??????? } catch (Throwable e) {
    ??????????? System.err.println(e);
    ??????? }
    ??? }
    }


    這個例子中,字段 d 的值被變為了 12.34。
    9.使用數組
    本文介紹的 reflection 的最后一種用法是創建的操作數組。數組在 Java 語言中是一種特殊的類類型,一個數組的引用可以賦給 Object 引用。觀察下面的例子看看數組是怎么工作的:
    import java.lang.reflect.*;
    public class array1 {
    ??? public static void main(String args[]) {
    ??????? try {
    ??????????? Class cls = Class.forName("java.lang.String");
    ??????????? Object arr = Array.newInstance(cls, 10);
    ??????????? Array.set(arr, 5, "this is a test");
    ??????????? String s = (String) Array.get(arr, 5);
    ??????????? System.out.println(s);
    ??????? } catch (Throwable e) {
    ??????????? System.err.println(e);
    ??????? }
    ??? }
    }


    例中創建了 10 個單位長度的 String 數組,為第 5 個位置的字符串賦了值,最后將這個字符串從數組中取得并打印了出來。

    下面這段代碼提供了一個更復雜的例子:
    import java.lang.reflect.*;
    public class array2 {
    ??? public static void main(String args[]) {
    ??????? int dims[] = new int[]{5, 10, 15};
    ??????? Object arr = Array.newInstance(Integer.TYPE, dims);
    ??????? Object arrobj = Array.get(arr, 3);
    ??????? Class cls = arrobj.getClass().getComponentType();
    ??????? System.out.println(cls);
    ??????? arrobj = Array.get(arrobj, 5);
    ??????? Array.setInt(arrobj, 10, 37);
    ??????? int arrcast[][][] = (int[][][]) arr;
    ??????? System.out.println(arrcast[3][5][10]);
    ??? }
    }
    例中創建了一個 5 x 10 x 15 的整型數組,并為處于 [3][5][10] 的元素賦了值為 37。注意,多維數組實際上就是數組的數組,例如,第一個 Array.get 之后,arrobj 是一個 10 x 15 的數組。進而取得其中的一個元素,即長度為 15 的數組,并使用 Array.setInt 為它的第 10 個元素賦值。

    注意創建數組時的類型是動態的,在編譯時并不知道其類型。

    posted @ 2006-06-05 11:05 nbt 閱讀(338) | 評論 (1)編輯 收藏

    Ruby On Railse的資源

    ?? Ruby On Railse官方網站:
    http://www.rubyonrails.org/
    可以作為了解這項技術的起點,那里面有一個15分鐘搭建一個blog的視頻很值得一看

    Rolling?with?Ruby?on?Rails:
    http://www.onlamp.com/pub/a/onlamp/2005/01/20/rails.html?page=1
    Curt?Hibbs寫的對于ror的學習過程,還沒來得及看,但是實在是太多地方reference了隨大流吧

    Really?Getting?Started?in?Rails
    http://www.slash7.com/articles/2005/01/24/really-getting-started-in-rails
    另一處比較著名的入門網站

    Ruby?on?Rails中文社區
    http://www.railscn.com/
    不知為啥,教育網上這個網站還要用代理而且好像還不能注冊?或者是我的代理的問題?

    Setting?up?a?Rails?Development?Environment?on?Windows?Using?Eclipse
    http://www.napcs.com/howto/railsonwindows.html
    這是一個step?by?step的在windows上利用eclipse搭建ruby?on?rails開發平臺的指南
    我自己就是參考的這個東東。

    Why's(poignant)?guide?to?ruby
    http://poignantguide.net/ruby/
    一個很有意思的ruby語言的入門介紹,作者還會自己畫小狐貍插圖,
    還在側邊欄里寫自己的價值觀,讓我們覺得老外還是蠻可愛的
    恩,應該可以用強烈推薦來形容,寓學于樂。

    RailsOnWindows:
    http://wiki.rubyonrails.com/rails/pages/RailsOnWindows
    教你怎樣在windows下利用Rails進行開發,其中包括Apache服務器的配置等等
    服務器的問題可以從這里找到答案。

    Ajax?on?Rails
    http://www.onlamp.com/pub/a/onlamp/2005/06/09/rails_ajax.html
    Ajax現在絕對是熱點,當然不可放過雖然對Ajax還僅僅是名詞和用戶體驗上的了解-___-

    GemRails
    http://wiki.rubyonrails.com/rails/pages/GemRails
    GemRails是rails的管理器,相當于linux系統中的包管理器
    可以自動上網安裝新的rails包,可以自動解決依賴,非常方便

    posted @ 2006-05-18 15:46 nbt 閱讀(329) | 評論 (0)編輯 收藏

    掌握Ajax- 第一部分:Ajax簡介(轉IBM開發者)

         摘要: Ajax 由 HTML 、 JavaScript? 技術、 DHTML 和 DOM 組成,這一杰出的方法可以將笨拙的 Web 界面轉化成交互性的 Ajax 應用程序。本文的作者是一位 Ajax 專家,他演示了這些技術如何協同工作 ...  閱讀全文

    posted @ 2006-05-18 13:51 nbt 閱讀(242) | 評論 (0)編輯 收藏

    僅列出標題
    共9頁: 上一頁 1 2 3 4 5 6 7 8 9 
    <2025年5月>
    27282930123
    45678910
    11121314151617
    18192021222324
    25262728293031
    1234567

    導航

    統計

    常用鏈接

    留言簿(3)

    隨筆分類

    隨筆檔案

    文章分類

    文章檔案

    相冊

    收藏夾

    Java技術網站

    友情鏈接

    國內一些開源網站

    最新隨筆

    搜索

    積分與排名

    最新評論

    閱讀排行榜

    評論排行榜

    主站蜘蛛池模板: 亚洲美女在线观看播放| 亚洲成a人片在线不卡一二三区| 91大神免费观看| 亚洲中文无码永久免| 亚洲国产V高清在线观看| 91精品全国免费观看青青| 亚洲国产综合精品| 日韩免费在线观看| 国产特黄特色的大片观看免费视频 | 亚洲综合精品伊人久久| 亚洲精品国产高清嫩草影院| 97国产在线公开免费观看| 国产亚洲精品AAAA片APP| 久久精品亚洲一区二区| 日韩精品免费电影| 永久免费A∨片在线观看| 亚洲老熟女五十路老熟女bbw| 亚洲人成网77777亚洲色| 免费看a级黄色片| 久艹视频在线免费观看| 色婷婷亚洲一区二区三区| 亚洲天堂视频在线观看| 国产免费爽爽视频免费可以看| 亚洲精品免费观看| 三年片在线观看免费观看大全中国| 亚洲精品韩国美女在线| 亚洲AV无码不卡在线观看下载| 69av免费观看| A级毛片高清免费视频在线播放| 亚洲AV无码国产剧情| 亚洲最大免费视频网| 亚洲精品制服丝袜四区| 免费看国产一级片| 18禁网站免费无遮挡无码中文| 91在线免费观看| 人成电影网在线观看免费| 国产AV旡码专区亚洲AV苍井空| 亚洲成亚洲乱码一二三四区软件| 无码不卡亚洲成?人片| 日本精品人妻无码免费大全 | 曰批全过程免费视频在线观看 |