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

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

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

    javaGrowing

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

    #

    正則表達(dá)式:

    正則表達(dá)式是一種可以用于模式匹配和替換的強(qiáng)有力的工具,一個(gè)正則表達(dá)式就是由普通的字符(例如字符 a 到 z)以及特殊字符(稱為元字符)組成的文字模式,它描述在查找文字主體時(shí)待匹配的一個(gè)或多個(gè)字符串。正則表達(dá)式作為一個(gè)模板,將某個(gè)字符模式與所搜索的字符串進(jìn)行匹配。

    正則表達(dá)式在字符數(shù)據(jù)處理中起著非常重要的作用,我們可以用正則表達(dá)式完成大部分的數(shù)據(jù)分析處理工作,如:判斷一個(gè)串是否是數(shù)字、是否是有效的Email地址,從海量的文字資料中提取有價(jià)值的數(shù)據(jù)等等,如果不使用正則表達(dá)式,那么實(shí)現(xiàn)的程序可能會(huì)很長(zhǎng),并且容易出錯(cuò)。對(duì)這點(diǎn)本人深有體會(huì),面對(duì)大量工具書(shū)電子檔資料的整理工作,如果不懂得應(yīng)用正則表達(dá)式來(lái)處理,那么將是很痛苦的一件事情,反之則將可以輕松地完成,獲得事半功倍的效果。

    由于本文目的是要介紹如何在JAVA里運(yùn)用正則表達(dá)式,因此對(duì)剛接觸正則表達(dá)式的讀者請(qǐng)參考有關(guān)資料,在此因篇幅有限不作介紹。





    回頁(yè)首


    JAVA對(duì)正則表達(dá)式的支持:

    在JDK1.3或之前的JDK版本中并沒(méi)有包含正則表達(dá)式庫(kù)可供JAVA程序員使用,之前我們一般都在使用第三方提供的正則表達(dá)式庫(kù),這些第三方庫(kù)中有源代碼開(kāi)放的,也有需付費(fèi)購(gòu)買的,而現(xiàn)時(shí)在JDK1.4的測(cè)試版中也已經(jīng)包含有正則表達(dá)式庫(kù)---java.util.regex。

    故此現(xiàn)在我們有很多面向JAVA的正則表達(dá)式庫(kù)可供選擇,以下我將介紹兩個(gè)較具代表性的 Jakarta-OROjava.util.regex,首先當(dāng)然是本人一直在用的 Jakarta-ORO:





    回頁(yè)首


    Jakarta-ORO正則表達(dá)式庫(kù)

    1.簡(jiǎn)介:

    Jakarta-ORO是最全面以及優(yōu)化得最好的正則表達(dá)式API之一,Jakarta-ORO庫(kù)以前叫做OROMatcher,是由Daniel F. Savarese編寫(xiě),后來(lái)他將其贈(zèng)與Jakarta Project,讀者可在Apache.org的網(wǎng)站 下載 該API包。

    許多源代碼開(kāi)放的正則表達(dá)式庫(kù)都是支持Perl5兼容的正則表達(dá)式語(yǔ)法,Jakarta-ORO正則表達(dá)式庫(kù)也不例外,他與Perl 5正則表達(dá)式完全兼容。

    2.對(duì)象與其方法:

    ★PatternCompiler對(duì)象:
    我們?cè)谑褂肑akarta-ORO API包時(shí),最先要做的是,創(chuàng)建一個(gè)Perl5Compiler類的實(shí)例,并把它賦值給PatternCompiler接口對(duì)象。Perl5Compiler是PatternCompiler接口的一個(gè)實(shí)現(xiàn),允許你把正則表達(dá)式編譯成用來(lái)匹配的Pattern對(duì)象。

    												
    												
    												
    																														PatternCompiler compiler=new Perl5Compiler();
    	PatternCompiler compiler=new Perl5Compiler();
    												
    												
    												
    										

    ★Pattern對(duì)象:
    要把所對(duì)應(yīng)的正則表達(dá)式編譯成Pattern對(duì)象,需要調(diào)用compiler對(duì)象的compile()方法,并在調(diào)用參數(shù)中指定正則表達(dá)式。舉個(gè)例子,你可以按照下面這種方式編譯正則表達(dá)式"s[ahkl]y":

    												
    												
    												
    																														 Pattern pattern=null;
     
    														
    																Pattern pattern=null;
    try {
    pattern=compiler.compile("s[ahkl]y ");
    } catch (MalformedPatternException e) {
    e.printStackTrace();
    }

    在默認(rèn)的情況下,編譯器會(huì)創(chuàng)建一個(gè)對(duì)大小寫(xiě)敏感的模式(pattern)。因此,上面代碼編譯得到的模式只匹配"say"、"shy"、 "sky"和"sly",但不匹配"Say"和"skY"。要?jiǎng)?chuàng)建一個(gè)大小寫(xiě)不敏感的模式,你應(yīng)該在調(diào)用編譯器的時(shí)候指定一個(gè)額外的參數(shù):
    pattern=compiler.compile("s[ahkl]y",Perl5Compiler.CASE_INSENSITIVE_MASK);

    Pattern對(duì)象創(chuàng)建好之后,就可以通過(guò)PatternMatcher類用該P(yáng)attern對(duì)象進(jìn)行模式匹配。

    ★PatternMatcher對(duì)象:

    PatternMatcher對(duì)象依據(jù)Pattern對(duì)象和字符串展開(kāi)匹配檢查。你要實(shí)例化一個(gè)Perl5Matcher類并把結(jié)果賦值給PatternMatcher接口。Perl5Matcher類是PatternMatcher接口的一個(gè)實(shí)現(xiàn),它根據(jù)Perl 5正則表達(dá)式語(yǔ)法進(jìn)行模式匹配:
    PatternMatcher matcher=new Perl5Matcher();

    PatternMatcher對(duì)象提供了多個(gè)方法進(jìn)行匹配操作,這些方法的第一個(gè)參數(shù)都是需要根據(jù)正則表達(dá)式進(jìn)行匹配的字符串:

    1. boolean matches(String input, Pattern pattern):當(dāng)要求輸入的字符串input和正則表達(dá)式pattern精確匹配時(shí)使用該方法。也就是說(shuō)當(dāng)正則表達(dá)式完整地描述輸入字符串時(shí)返回真值。
    2. boolean matchesPrefix(String input, Pattern pattern):要求正則表達(dá)式匹配輸入字符串起始部分時(shí)使用該方法。也就是說(shuō)當(dāng)輸入字符串的起始部分與正則表達(dá)式匹配時(shí)返回真值。
    3. boolean contains(String input, Pattern pattern):當(dāng)正則表達(dá)式要匹配輸入字符串的一部分時(shí)使用該方法。當(dāng)正則表達(dá)式為輸入字符串的子串時(shí)返回真值。

    但以上三種方法只會(huì)查找輸入字符串中匹配正則表達(dá)式的第一個(gè)對(duì)象,如果當(dāng)字符串可能有多個(gè)子串匹配給定的正則表達(dá)式時(shí),那么你就可以在調(diào)用上面三個(gè)方法時(shí)用PatternMatcherInput對(duì)象作為參數(shù)替代String對(duì)象,這樣就可以從字符串中最后一次匹配的位置開(kāi)始繼續(xù)進(jìn)行匹配,這樣就方便的多了。

    用PatternMatcherInput對(duì)象作為參數(shù)替代String時(shí),上述三個(gè)方法的語(yǔ)法如下:

    1. boolean matches(PatternMatcherInput input, Pattern pattern)
    2. boolean matchesPrefix(PatternMatcherInput input, Pattern pattern)
    3. boolean contains(PatternMatcherInput input, Pattern pattern)

    ★Util.substitute()方法:
    查找后需要要進(jìn)行替換,我們就要用到Util.substitute()方法,其語(yǔ)法如下:

    												
    												
    												
    														
    																public static String substitute(PatternMatcher matcher,
    Pattern pattern,Substitution sub,String input,
    int numSubs)

    前兩個(gè)參數(shù)分別為PatternMatcher和Pattern對(duì)象。而第三個(gè)參數(shù)是個(gè)Substiution對(duì)象,由它來(lái)決定替換操作如何進(jìn)行。第四個(gè)參數(shù)是要進(jìn)行替換操作的目標(biāo)字符串,最后一個(gè)參數(shù)用來(lái)指定是否替換模式的所有匹配子串(Util.SUBSTITUTE_ALL),或只進(jìn)行指定次數(shù)的替換。

    在這里我相信有必要詳細(xì)解說(shuō)一下第三個(gè)參數(shù)Substiution對(duì)象,因?yàn)樗鼘Q定替換將怎樣進(jìn)行。

    Substiution:
    Substiution是一個(gè)接口類,它為你提供了在使用Util.substitute()方法時(shí)控制替換方式的手段,它有兩個(gè)標(biāo)準(zhǔn)的實(shí)現(xiàn)類:StringSubstitution與Perl5Substitution。當(dāng)然,同時(shí)你也可以生成自己的實(shí)現(xiàn)類來(lái)定制你所需要的特殊替換動(dòng)作。

    StringSubstitution:
    StringSubstitution 實(shí)現(xiàn)的是簡(jiǎn)單的純文字替換手段,它有兩個(gè)構(gòu)造方法:

    StringSubstitution()->缺省的構(gòu)造方法,初始化一個(gè)包含零長(zhǎng)度字符串的替換對(duì)象。

    StringSubstitution(java.lang.String substitution)->初始化一個(gè)給定字符串的替換對(duì)象。

    Perl5Substitution:
    Perl5Substitution 是StringSubstitution的子類,它在實(shí)現(xiàn)純文字替換手段的同時(shí)也允許進(jìn)行針對(duì)MATH類里各匹配組的PERL5變量的替換,所以他的替換手段比其直接父類StringSubstitution更為多元化。

    它有三個(gè)構(gòu)造器:

    Perl5Substitution()

    Perl5Substitution(java.lang.String substitution)

    Perl5Substitution(java.lang.String substitution, int numInterpolations)

    前兩種構(gòu)造方法與StringSubstitution一樣,而第三種構(gòu)造方法下面將會(huì)介紹到。

    Perl5Substitution的替換字符串中可以包含用來(lái)替代在正則表達(dá)式里由小擴(kuò)號(hào)圍起來(lái)的匹配組的變量,這些變量是由$1, $2,$3等形式來(lái)標(biāo)識(shí)。我們可以用一個(gè)例子來(lái)解釋怎樣使用替換變量來(lái)進(jìn)行替換:

    假設(shè)我們有正則表達(dá)式模式為b\d+:(也就是b[0-9]+:),而我們想把所有匹配的字符串中的"b"都改為"a",而":"則改為"-",而其余部分則不作修改,如我們輸入字符串為"EXAMPLE b123:",經(jīng)過(guò)替換后就應(yīng)該變成"EXAMPLE a123-"。要做到這點(diǎn),我們就首先要把不做替換的部分用分組符號(hào)小括號(hào)包起來(lái),這樣正則表達(dá)式就變?yōu)?b(\d+):",而構(gòu)造Perl5Substitution對(duì)象時(shí)其替換字符串就應(yīng)該是"a$1-",也就是構(gòu)造式為Perl5Substitution("a$1-"),表示在使用Util.substitute()方法時(shí)只要在目標(biāo)字符串里找到和正則表達(dá)式" b(\d+): "相匹配的子串都用替換字符串來(lái)替換,而變量$1表示如果和正則表達(dá)式里第一個(gè)組相匹配的內(nèi)容則照般原文插到$1所在的為置,如在"EXAMPLE b123:"中和正則表達(dá)式相匹配的部分是"b123:",而其中和第一分組"(\d+)"相匹配的部分則是"123",所以最后替換結(jié)果為"EXAMPLE a123-"。

    有一點(diǎn)需要清楚的是,如果你把構(gòu)造器Perl5Substitution(java.lang.String substitution,int numInterpolations)

    中的numInterpolations參數(shù)設(shè)為INTERPOLATE_ALL,那么當(dāng)每次找到一個(gè)匹配字串時(shí),替換變量($1,$2等)所指向的內(nèi)容都根據(jù)目前匹配字串來(lái)更新,但是如果numInterpolations參數(shù)設(shè)為一個(gè)正整數(shù)N時(shí),那么在替換時(shí)就只會(huì)在前N次匹配發(fā)生時(shí)替換變量會(huì)跟隨匹配對(duì)象來(lái)調(diào)整所代表的內(nèi)容,但N次之后就以一致以第N次替換變量所代表內(nèi)容來(lái)做為以后替換結(jié)果。

    舉個(gè)例子會(huì)更好理解:

    假如沿用以上例子中的正則表達(dá)式模式以及替換內(nèi)容來(lái)進(jìn)行替換工作,設(shè)目標(biāo)字符串為"Tank b123: 85 Tank b256: 32 Tank b78: 22",并且設(shè)numInterpolations參數(shù)為INTERPOLATE_ALL,而Util.substitute()方法中的numSub變量設(shè)為SUBSTITUTE_ALL(請(qǐng)參考上文Util.substitute()方法內(nèi)容),那么你獲得的替換結(jié)果將會(huì)是: Tank a123- 85 Tank a256- 32 Tank a78- 22

    但是如果你把numInterpolations設(shè)為2,并且numSubs依然設(shè)為SUBSTITUTE_ALL,那么這時(shí)你獲得的結(jié)果則會(huì)是: Tank a123- 85 Tank a256- 32 Tank a256- 22

    你要注意到最后一個(gè)替換所用變量$1所代表的內(nèi)容與第二個(gè)$1一樣為"256",而不是預(yù)期的"78",因?yàn)樵谔鎿Q進(jìn)行中,替換變量$1只根據(jù)匹配內(nèi)容進(jìn)行了兩次更新,最后一次就使第二次匹配時(shí)所更新的結(jié)果,那么我們可以由此知道,如果numInterpolations設(shè)為1,那么結(jié)果將是: Tank a123- 85 Tank a123- 32 Tank a123- 22

    3.應(yīng)用示例:

    剛好前段時(shí)間公司準(zhǔn)備出一個(gè)《伊索預(yù)言》的英語(yǔ)學(xué)習(xí)互動(dòng)教材,其中有電子檔資料的整理工作,我們就以此為例來(lái)看一下Jakarta-ORO與JDBC2.0 API結(jié)合起來(lái)對(duì)數(shù)據(jù)庫(kù)內(nèi)的資料進(jìn)行簡(jiǎn)單提取與整理的實(shí)現(xiàn)。假設(shè)由錄入部的同事送過(guò)來(lái)的存放在MS SQLSERVER 7數(shù)據(jù)庫(kù)里的電子檔的表結(jié)構(gòu)如下(注:或許在不同的DBMS中有相應(yīng)的正則表達(dá)式的應(yīng)用,但這不在本文討論范圍內(nèi)):

    表名:AESOP, 表中每條記錄包含有三列:
    ID(int):?jiǎn)卧~索引號(hào)
    WORD(varchar):?jiǎn)卧~
    CONTENT(varchar):存放單詞的相關(guān)解釋與例句等內(nèi)容

    其中CONTENT列中內(nèi)容的格式如下:
    [音標(biāo)] [詞性] (解釋){(例句一/例句解釋/例句中該詞的詞性: 單詞在句中的意思) (例句二/例句解釋/例句中該詞的詞性: 單詞在句中的意思)}

    如對(duì)應(yīng)單詞Kevin,CONTENT中的內(nèi)容如下:
    ['kevin] [名詞](人名凱文){(Kevin loves comic./凱文愛(ài)漫畫(huà)/名詞: 凱文)( Kevin is living in ZhuHai now./凱文現(xiàn)住在珠海/名詞: 凱文)}

    我們的例子主要針對(duì)CONTENT列中內(nèi)容進(jìn)行字符串處理。

    ★查找單個(gè)匹配:
    首先,讓我們嘗試把CONTNET列中的[音標(biāo)]字段的內(nèi)容列示出來(lái),由于所有單詞的記錄中都有這一項(xiàng)并且都在字串開(kāi)始位置,所以這個(gè)查找工作比較簡(jiǎn)單:

    1. 確定相應(yīng)的正則表達(dá)式:\[[^]]+\]

      這個(gè)是很簡(jiǎn)單的正則表達(dá)式,其意思是要求相匹配的字符串必須為以一對(duì)中括號(hào)包含的所有內(nèi)容,如['kevin] 、[名詞]等,但內(nèi)容中不包括"]"符號(hào),也就是要避免出現(xiàn)"[][]"會(huì)作為一個(gè)匹配對(duì)象的情況出現(xiàn)(有關(guān)正則表達(dá)式的基礎(chǔ)知識(shí)請(qǐng)參照有關(guān)資料,這里不再詳述)。

      注意,在Java中,你必須對(duì)每一個(gè)向前的斜杠("\")進(jìn)行轉(zhuǎn)義處理。所以我們要在上面的正則表達(dá)式里每個(gè)"\"前面加上一個(gè)"\"以免出現(xiàn)編譯錯(cuò)誤,也就是在JAVA中初始化正則表達(dá)式的字符串的語(yǔ)句應(yīng)該為:

      String restring=" \\[[^]]+\\]";

      并且在表達(dá)式里每個(gè)符號(hào)中間不能有空格,否則就會(huì)同樣出現(xiàn)編譯錯(cuò)誤。

    2. 實(shí)例化PatternCompiler對(duì)象,創(chuàng)建Pattern對(duì)象

      PatternCompiler compiler=new Perl5Compiler();

      Pattern pattern=compiler.compile(restring);

    3. 創(chuàng)建PatternMatcher對(duì)象,調(diào)用PatternMatcher接口的contain()方法檢查匹配情況:
      																
      																
      																
      																																						  PatternMatcher matcher=new Perl5Matcher();
              if (matcher.contains(content,pattern)) {
                       //處理代碼片段
              }
      
      																		
      																
      																
      																
      														

      這里matcher.contains(content,pattern)中的參數(shù) content是從數(shù)據(jù)庫(kù)里取來(lái)的字符串變量。該方法只會(huì)查到第一個(gè)匹配的對(duì)象字符串,但是由于音標(biāo)項(xiàng)均在CONETNET內(nèi)容字符串中的起始位置,所以用這個(gè)方法就已經(jīng)可以保證把每條記錄里的音標(biāo)項(xiàng)找出來(lái)了,但更為直接與合理的辦法是使用boolean matchesPrefix(PatternMatcherInput input, Pattern pattern)方法,該方法驗(yàn)證目標(biāo)字符串是否以正則表達(dá)式所匹配的字串為起始。

      具體實(shí)現(xiàn)的完整的程序代碼如下:

      																
      																
      																
      																																						package RegularExpressions;
      //import……
      import org.apache.oro.text.regex.*;
      //使用Jakarta-ORO正則表達(dá)式庫(kù)前需要把它加到CLASSPATH里面,如果用IDE是//JBUILDER,那么也可以在JBUILDER里直接自建新庫(kù)。
      
      public class yisuo{
        public static void main(String[] args){
        try{     
      //使用JDBC DRIVER進(jìn)行DBMS連接,這里我使用的是一個(gè)第三方JDBC 
      //DRIVER,Microsoft本身也有一個(gè)面向SQLSERVER7/2000的免費(fèi)JDBC //DRIVER,但其性能真的是奇差,不用也罷。
              Class.forName("com.jnetdirect.jsql.JSQLDriver");
                Connection con=DriverManager.getConnection
                ("jdbc:JSQLConnect://kevin:1433","kevin chen","re");
                Statement stmt = con.createStatement(ResultSet.TYPE_SCROLL_SENSITIVE,
                ResultSet.CONCUR_UPDATABLE);
      //為使用Jakarta-ORO庫(kù)而創(chuàng)建相應(yīng)的對(duì)象
      String rsstring=" \\[[^]]+\\]"; 
                PatternCompiler orocom=new Perl5Compiler();
                Pattern pattern=orocom.compile(rsstring);
                PatternMatcher matcher=new Perl5Matcher();
                ResultSet uprs = stmt.executeQuery("SELECT * FROM aesop");
                while (uprs.next()) {
      Stirng  word=uprs.getString("word");
                Stirng  content=uprs.getString("content");
                  if(matcher.contains(content,pattern)){
                //或if(matcher.matchesPrefix(content,pattern)){
                      MatchResult result=matcher.getMatch();
                      Stirng pure=result.toString();
                      System.out.println(word+"的音標(biāo)為:"+pure);
                  }
                }
             }
        catch(Exception e) {
                   System.out.println(e);
             }
        }
      }
      																		
      																
      																
      																
      														

      輸出結(jié)果為:kevin的音標(biāo)為['kevin]

    在這個(gè)處理中我是用toString()方法來(lái)取得結(jié)果,但是如果正則表達(dá)式里是用了分組符號(hào)(圓括號(hào)),那么就可以用group(int gid)的方法來(lái)取得相應(yīng)各組匹配的結(jié)果,如正則表達(dá)式改為" (\[[^]]+\])",那么就可以用以下方法來(lái)取得結(jié)果:pure=result.group(0);

    用程序驗(yàn)證,輸出結(jié)果同樣為:kevin的音標(biāo)為['kevin]

    而如果正則表達(dá)式為(\[[^]]+\])(\[[^]]+\]),則會(huì)查找到兩個(gè)連續(xù)的方括號(hào)所包含的內(nèi)容,也就找到[音標(biāo)] [詞性]兩項(xiàng),但是兩項(xiàng)的結(jié)果分別在兩個(gè)組里面,分別由下面語(yǔ)句獲得結(jié)果:

    result.group(0)->返回[音標(biāo)] [詞性]兩項(xiàng)內(nèi)容,也就是與整個(gè)正則表達(dá)式相匹配的結(jié)果字符串,在這里也就為['kevin] [名詞]

    result.group(1) ->返回[音標(biāo)]項(xiàng)內(nèi)容,結(jié)果應(yīng)是['kevin]

    result.group(2) ->返回[詞性]項(xiàng)內(nèi)容,結(jié)果應(yīng)是[名詞]

    繼續(xù)用程序驗(yàn)證,發(fā)現(xiàn)輸出并不正確,主要是當(dāng)內(nèi)容有中文時(shí)就不能成功匹配,考慮到可能是Jakarta-ORO正則表達(dá)式庫(kù)版本不支持中文的問(wèn)題,回看一下原來(lái)我一直用的還是2.0.1的老版本,馬上到Jakarta.org上下載最新的2.0.4版本裝上再用程序驗(yàn)證,得出的結(jié)果就和預(yù)期一樣正確。

    ★查找多個(gè)匹配:
    經(jīng)過(guò)第一步的嘗試使用Jakarta-ORO后,我們已經(jīng)知道了如何正確使用該API包來(lái)查找目標(biāo)字符串里一個(gè)匹配的子串,下面我們接著來(lái)看一看當(dāng)目標(biāo)字符串里包含不止一個(gè)匹配的子串時(shí)我們?nèi)绾伟阉鼈円粋€(gè)接一個(gè)找出來(lái)進(jìn)行相應(yīng)的處理。

    首先我們先試個(gè)簡(jiǎn)單的應(yīng)用,假設(shè)我們想把CONTNET字段內(nèi)容里所有用方括號(hào)包起來(lái)的字串都找出來(lái),很清楚地,CONTNET字段的內(nèi)容里面就只有兩項(xiàng)匹配的內(nèi)容:[音標(biāo)]和 [詞性],剛才我們其實(shí)已經(jīng)把它們分別找出來(lái)了,但是我們所用的方法是分組方法,把"[音標(biāo)] [詞性]"作為一整個(gè)正則表達(dá)式匹配的內(nèi)容先找到,再根據(jù)分組把[音標(biāo)]和 [詞性]分別挑出來(lái)。但是現(xiàn)在我們需要做的是把[音標(biāo)]和[詞性]分別做為與同一個(gè)正則表達(dá)式匹配的內(nèi)容,先找到一個(gè)接著再找下一個(gè),也就是剛才我們的表達(dá)式為(\[[^]]+\])(\[[^]]+\]),而現(xiàn)在應(yīng)為" \[[^]]+\] "。

    我們已經(jīng)知道在匹配操作的三個(gè)方法里只要用PatternMatcherInput對(duì)象作為參數(shù)替代String對(duì)象就可以從字符串中最后一次匹配的位置開(kāi)始繼續(xù)進(jìn)行匹配,實(shí)現(xiàn)的程序片段如下:

    												
    												
    												
    																														PatternMatcherInput input=new PatternMatcherInput(content);
                while (matcher.contains(input,pattern)) {
                    result=matcher.getMatch();
                    System.out.println(result.group(0)) 
                }
    
    														
    												
    												
    												
    										

    輸出結(jié)果為:['kevin]
    [名詞]

    接著我們來(lái)做復(fù)雜一點(diǎn)的處理,就是我們要先把下面內(nèi)容:
    ['kevin] [名詞](人名凱文){(Kevin loves comic./凱文愛(ài)漫畫(huà)/名詞: 凱文)( Kevin is living in ZhuHai now. /凱文現(xiàn)住在珠海/名詞: 凱文)}中的整個(gè)例句部分(也就是由大括號(hào)所包含的部分)找出來(lái),再分別把例句一和例句二找出,而各例句中的各項(xiàng)內(nèi)容(英文句、中文句、詞性、解釋)也要分項(xiàng)列出。

    第一步當(dāng)然是要定出相應(yīng)的正則表達(dá)式,需要有兩個(gè),一是和整個(gè)例句部分(也就是由大括號(hào)包起來(lái)的部分)匹配的正則表達(dá)式:"\{.+\}",

    另一個(gè)則要和每個(gè)例句部分匹配(也就是小括號(hào)中的內(nèi)容),:\(([^)]+\)


    而且由于要把例句的各項(xiàng)分離出來(lái),所以要再把里面的各部分用分組的方法匹配出來(lái):" ([^(]+)/(.+)/(.+):([^)]+) "。

    為了簡(jiǎn)便起見(jiàn),我們不再和從數(shù)據(jù)庫(kù)里讀出,而是構(gòu)造一個(gè)包含同樣內(nèi)容的字符串變量,程序片段如下:

    												
    														try{
             String content="['kevin] [名詞](人名凱文){(Kevin loves comic./凱文愛(ài)漫畫(huà)/名詞:凱文) (Kevin is living in ZhuHai now./凱文現(xiàn)住在珠海/名詞: 凱文)}";
             String ps1="\\{.+\\}";
             String ps2="\\([^)]+\\)";
             String ps3="([^(]+)/(.+)/(.+):([^)]+)";
             String sentence;
             PatternCompiler orocom=new Perl5Compiler();
             Pattern pattern1=orocom.compile(ps1);
             Pattern pattern2=orocom.compile(ps2);
             Pattern pattern3=orocom.compile(ps3);
             PatternMatcher matcher=new Perl5Matcher();
    //先找出整個(gè)例句部分
                if (matcher.contains(content,pattern1)) {
                MatchResult result=matcher.getMatch();
                String example=result.toString();
                PatternMatcherInput input=new PatternMatcherInput(example);
            //分別找出例句一和例句二
                while (matcher.contains(input,pattern2)){
                    result=matcher.getMatch();
                    sentence=result.toString();
            //把每個(gè)例句里的各項(xiàng)用分組的辦法分隔出來(lái)
                    if (matcher.contains(sentence,pattern3)){
                      result=matcher.getMatch();
                      System.out.println("英文句: "+result.group(1));
                      System.out.println("句子中文翻譯: "+result.group(2));
                      System.out.println("詞性: "+result.group(3));
                      System.out.println("意思: "+result.group(4));
                    }
                }
            }
           }
      catch(Exception e) {
                 System.out.println(e);
           }
    
    												
    										

    輸出結(jié)果為:
    英文句: Kevin loves comic.
    句子中文翻譯: 凱文愛(ài)漫畫(huà)
    詞性: 名詞
    意思: 凱文
    英文句: Kevin is living in ZhuHai now.
    句子中文翻譯: 凱文現(xiàn)住在珠海
    詞性: 名詞
    意思: 凱文

    ★查找替換:
    以上的兩個(gè)應(yīng)用都是單純?cè)诓檎易址ヅ浞矫娴模覀冊(cè)賮?lái)看一下查找后如何對(duì)目標(biāo)字符串進(jìn)行替換。

    例如我現(xiàn)在想把第二個(gè)例句進(jìn)行改動(dòng),換為:Kevin has seen《LEON》seveal times,because it is a good film./ 凱文已經(jīng)看過(guò)《這個(gè)殺手不太冷》幾次了,因?yàn)樗且徊亢秒娪啊?名詞:凱文。

    也就是把
    ['kevin] [名詞](人名凱文){(Kevin loves comic./凱文愛(ài)漫畫(huà)/名詞: 凱文)( Kevin is living in ZhuHai now. /凱文現(xiàn)住在珠海/名詞: 凱文)}

    改為:
    ['kevin] [名詞](人名凱文){(Kevin loves comic./凱文愛(ài)漫畫(huà)/名詞: 凱文)( Kevin has seen《LEON》seveal times,because it is a good film./ 凱文已經(jīng)看過(guò)《這個(gè)殺手不太冷》幾次了,因?yàn)樗且徊亢秒娪啊?名詞:凱文。)}

    之前,我們已經(jīng)了解Util.substitute()方法與Substiution接口,以及Substiution的兩個(gè)實(shí)現(xiàn)類StringSubstitution和Perl5Substitution,我們就來(lái)看看怎么用Util.substitute()方法配合Perl5Substitution來(lái)完成我們上面提出的替換要求,確定正則表達(dá)式:

    我們要先找到其中的整個(gè)例句部分,也就是由大括號(hào)包起來(lái)的字串,并且把兩個(gè)例句分別分組,所以正則表達(dá)式為:"\{(\([^)]+\))(\([^)]+\))\}",如果用替換變量來(lái)代替分組,那么上面的表達(dá)式可以看為"\{$1$2\}",這樣就可以更容易看出替換變量與分組間的關(guān)系。

    根據(jù)上面的正則表達(dá)式Perl5Substitution類可以這樣構(gòu)造: Perl5Substitution("{$1( Kevin has seen《LEON》seveal times,because it is a good film./ 凱文已經(jīng)看過(guò)《這個(gè)殺手不太冷》幾次了,因?yàn)樗且徊亢秒娪啊?名詞:凱文。)}")

    再根據(jù)這個(gè)Perl5Substitution對(duì)象來(lái)使用Util.substitute()方法便可以完成替換了,實(shí)現(xiàn)的代碼片段如下:

    												
    														try{
       String content="['kevin] [名詞](人名凱文){(Kevin loves comic.
       /凱文愛(ài)漫畫(huà)/名詞: 凱文)(Kevin lives in ZhuHai now./凱文現(xiàn)住在珠海/名詞: 凱文)}";
       String ps1="\\{(\\([^)]+\\))(\\([^)]+\\))\\}";
       String sentence;
       String pure;
       PatternCompiler orocom=new Perl5Compiler();
       Pattern pattern1=orocom.compile(ps1);
       PatternMatcher matcher=new Perl5Matcher();
           String result=Util.substitute(matcher,
            pattern1,new Perl5Substitution(
           "{$1( Kevin has seen《LEON》seveal times,because it is a good film./ 
           凱文已經(jīng)看過(guò)《這個(gè)殺手不太冷》幾次了,因?yàn)樗且徊亢秒娪啊?名詞:凱文。)}",1),
            content,Util.SUBSTITUTE_ALL);
            System.out.println(result);
       }
      catch(Exception e) {
                 System.out.println(e);
           }
    
    												
    										

    輸出結(jié)果是正確的,為:
    ['kevin] [名詞](人名凱文){(Kevin loves comic./凱文愛(ài)漫畫(huà)/名詞: 凱文)( Kevin has seen《LEON》seveal times,because it is a good film./ 凱文已經(jīng)看過(guò)《這個(gè)殺手不太冷》幾次了,因?yàn)樗且徊亢秒娪啊?名詞:凱文。)}

    至于有關(guān)使用numInterpolations參數(shù)的構(gòu)造器用法,讀者只要根據(jù)上面的介紹自己動(dòng)手試一下就會(huì)清楚了,在此就不再例述。





    回頁(yè)首


    總結(jié):

    本文首先介紹了Jakarta-ORO正則表達(dá)式庫(kù)的對(duì)象與方法,并且接著舉例讓讀者對(duì)實(shí)際應(yīng)用有進(jìn)一步的了解,雖然例子都比較簡(jiǎn)單,但希望讀者們?cè)诳戳嗽撐暮髮?duì)Jakarta-ORO正則表達(dá)式庫(kù)有一定的認(rèn)知,在實(shí)際工作中有所幫助與啟發(fā)。

    其實(shí)在Jakarta org里除了Jakarta-ORO外還有一個(gè)百分百的純JAVA正則表達(dá)式庫(kù),就是由Jonathan Locke贈(zèng)與Jakarta ORG的Regexp,在該包里面包含了完整的文檔以及一個(gè)用于調(diào)試的Applet例子,對(duì)其有興趣的讀者可以到此 下載

    posted @ 2006-05-22 13:19 javaGrowing 閱讀(1225) | 評(píng)論 (0)編輯 收藏

    <?xml version="1.0" encoding="UTF-8" ?>
    <!DOCTYPE log4j:configuration SYSTEM "log4j.dtd">
    <log4j:configuration xmlns:log4j='http://jakarta.apache.org/log4j/'>

    <appender name="STDOUT" class="org.apache.log4j.ConsoleAppender">
    <layout class="org.apache.log4j.PatternLayout">
    <param name="ConversionPattern" value="%c %d{ISO8601}
    -- %p -- %m%n"/>
    </layout>
    </appender>

    <appender name="DEBUG" class="org.apache.log4j.RollingFileAppender">
    <param name="File" value="C:\\logs\\app-debug.log"/>
    <param name="Append" value="true"/>
    <param name="MaxFileSize" value="500KB"/>
    <param name="MaxBackupIndex" value="2"/>
    <layout class="org.apache.log4j.PatternLayout">
    <param name="ConversionPattern" value="%c %d{ISO8601}
    -- %p -- %m%n"/>
    </layout>
    <filter class="org.apache.log4j.varia.LevelRangeFilter">
    <param name="LevelMin" value="DEBUG" />
    <param name="LevelMax" value="DEBUG" />
    </filter>
    </appender>

    <appender name="INFO" class="org.apache.log4j.RollingFileAppender">
    <param name="File" value="C:\\logs\\app-info.log"/>
    <param name="Append" value="true"/>
    <param name="MaxFileSize" value="500KB"/>
    <param name="MaxBackupIndex" value="2"/>
    <layout class="org.apache.log4j.PatternLayout">
    <param name="ConversionPattern" value="%c %d{ISO8601}
    -- %p -- %m%n"/>
    </layout>
    <filter class="org.apache.log4j.varia.LevelRangeFilter">
    <param name="LevelMin" value="INFO" />
    <param name="LevelMax" value="INFO" />
    </filter>
    </appender>

    <appender name="WARN" class="org.apache.log4j.RollingFileAppender">
    <param name="File" value="C:\\logs\\app-warn.log"/>
    <param name="Append" value="true"/>
    <param name="MaxFileSize" value="500KB"/>
    <param name="MaxBackupIndex" value="2"/>
    <layout class="org.apache.log4j.PatternLayout">
    <param name="ConversionPattern" value="%c %d{ISO8601}
    -- %p -- %m%n"/>
    </layout>
    <filter class="org.apache.log4j.varia.LevelRangeFilter">
    <param name="LevelMin" value="WARN" />
    <param name="LevelMax" value="WARN" />
    </filter>
    </appender>

    <appender name="ERROR" class="org.apache.log4j.RollingFileAppender">
    <param name="File" value="C:\\logs\\app-error.log"/>
    <param name="Append" value="true"/>
    <param name="MaxFileSize" value="500KB"/>
    <param name="MaxBackupIndex" value="2"/>
    <layout class="org.apache.log4j.PatternLayout">
    <param name="ConversionPattern" value="%c %d{ISO8601}
    -- %p -- %m%n"/>
    </layout>
    <filter class="org.apache.log4j.varia.LevelRangeFilter">
    <param name="LevelMin" value="ERROR" />
    <param name="LevelMax" value="ERROR" />
    </filter>
    </appender>

    <root>
    <priority value="debug"/>
    <appender-ref ref="STDOUT"/>
    <appender-ref ref="DEBUG"/>
    <appender-ref ref="INFO"/>
    <appender-ref ref="WARN"/>
    <appender-ref ref="ERROR"/>
    </root>

    </log4j:configuration>
    posted @ 2006-05-15 10:04 javaGrowing 閱讀(732) | 評(píng)論 (1)編輯 收藏

    編者按:現(xiàn)在開(kāi)發(fā)Java Web應(yīng)用,建立和部署Web內(nèi)容是一件很簡(jiǎn)單的工作。使用Jakarta Tomcat作為Servlet和JSP容器的人已經(jīng)遍及全世界。Tomcat具有免費(fèi)、跨平臺(tái)等諸多特性,并且更新得很快,現(xiàn)在非常的流行。



    你所需要做的就是:按照你的需求配置Tomcat,只要你正確配置,Tomcat一般都能適合你的要求。下面是一系列關(guān)于Tomcat的配置技巧,這些技巧源自于我的書(shū):《Tomcat權(quán)威指南》,希望對(duì)你有所幫助。—— Jason Brittain



    1. 配置系統(tǒng)管理(Admin Web Application)

    大多數(shù)商業(yè)化的J2EE服務(wù)器都提供一個(gè)功能強(qiáng)大的管理界面,且大都采用易于理解的Web應(yīng)用界面。Tomcat按照自己的方式,同樣提供一個(gè)成熟的管理工具,并且絲毫不遜于那些商業(yè)化的競(jìng)爭(zhēng)對(duì)手。Tomcat的Admin Web Application最初在4.1版本時(shí)出現(xiàn),當(dāng)時(shí)的功能包括管理context、data source、user和group等。當(dāng)然也可以管理像初始化參數(shù),user、group、role的多種數(shù)據(jù)庫(kù)管理等。在后續(xù)的版本中,這些功能將得到很大的擴(kuò)展,但現(xiàn)有的功能已經(jīng)非常實(shí)用了。



    Admin Web Application被定義在自動(dòng)部署文件:CATALINA_BASE/webapps/admin.xml 。

    (譯者注:CATALINA_BASE即tomcat安裝目錄下的server目錄)



    你必須編輯這個(gè)文件,以確定Context中的docBase參數(shù)是絕對(duì)路徑。也就是說(shuō),CATALINA_BASE/webapps/admin.xml 的路徑是絕對(duì)路徑。作為另外一種選擇,你也可以刪除這個(gè)自動(dòng)部署文件,而在server.xml文件中建立一個(gè)Admin Web Application的context,效果是一樣的。你不能管理Admin Web Application這個(gè)應(yīng)用,換而言之,除了刪除CATALINA_BASE/webapps/admin.xml ,你可能什么都做不了。



    如果你使用UserDatabaseRealm(默認(rèn)),你將需要添加一個(gè)user以及一個(gè)role到CATALINA_BASE/conf/tomcat-users.xml 文件中。你編輯這個(gè)文件,添加一個(gè)名叫“admin”的role 到該文件中,如下:



    <role name="admin"/>



    你同樣需要有一個(gè)用戶,并且這個(gè)用戶的角色是“admin”。象存在的用戶那樣,添加一個(gè)用戶(改變密碼使其更加安全):



    <user name="admin" password="deep_dark_secret" roles="admin"/>



    當(dāng)你完成這些步驟后,請(qǐng)重新啟動(dòng)Tomcat,訪問(wèn)http://localhost:8080/admin,你將看到一個(gè)登錄界面。Admin Web Application采用基于容器管理的安全機(jī)制,并采用了Jakarta Struts框架。一旦你作為“admin”角色的用戶登錄管理界面,你將能夠使用這個(gè)管理界面配置Tomcat。



    2.配置應(yīng)用管理(Manager Web Application)

    Manager Web Application讓你通過(guò)一個(gè)比Admin Web Application更為簡(jiǎn)單的用戶界面,執(zhí)行一些簡(jiǎn)單的Web應(yīng)用任務(wù)。



    Manager Web Application被被定義在一個(gè)自動(dòng)部署文件中:



    CATALINA_BASE/webapps/manager.xml 。



    你必須編輯這個(gè)文件,以確保context的docBase參數(shù)是絕對(duì)路徑,也就是說(shuō)CATALINA_HOME/server/webapps/manager的絕對(duì)路徑。

    (譯者注:CATALINA_HOME即tomcat安裝目錄)



    如果你使用的是UserDatabaseRealm,那么你需要添加一個(gè)角色和一個(gè)用戶到CATALINA_BASE/conf/tomcat-users.xml文件中。接下來(lái),編輯這個(gè)文件,添加一個(gè)名為“manager”的角色到該文件中:



    <role name=”manager”>



    你同樣需要有一個(gè)角色為“manager”的用戶。像已經(jīng)存在的用戶那樣,添加一個(gè)新用戶(改變密碼使其更加安全):



    <user name="manager" password="deep_dark_secret" roles="manager"/>



    然后重新啟動(dòng)Tomcat,訪問(wèn)http://localhost/manager/list,將看到一個(gè)很樸素的文本型管理界面,或者訪問(wèn)http://localhost/manager/html/list,將看到一個(gè)HMTL的管理界面。不管是哪種方式都說(shuō)明你的Manager Web Application現(xiàn)在已經(jīng)啟動(dòng)了。



    Manager application讓你可以在沒(méi)有系統(tǒng)管理特權(quán)的基礎(chǔ)上,安裝新的Web應(yīng)用,以用于測(cè)試。如果我們有一個(gè)新的web應(yīng)用位于/home/user/hello下在,并且想把它安裝到 /hello下,為了測(cè)試這個(gè)應(yīng)用,我們可以這么做,在第一個(gè)文件框中輸入“/hello”(作為訪問(wèn)時(shí)的path),在第二個(gè)文本框中輸入“file:/home/user/hello”(作為Config URL)。



    Manager application還允許你停止、重新啟動(dòng)、移除以及重新部署一個(gè)web應(yīng)用。停止一個(gè)應(yīng)用使其無(wú)法被訪問(wèn),當(dāng)有用戶嘗試訪問(wèn)這個(gè)被停止的應(yīng)用時(shí),將看到一個(gè)503的錯(cuò)誤——“503 - This application is not currently available”。



    移除一個(gè)web應(yīng)用,只是指從Tomcat的運(yùn)行拷貝中刪除了該應(yīng)用,如果你重新啟動(dòng)Tomcat,被刪除的應(yīng)用將再次出現(xiàn)(也就是說(shuō),移除并不是指從硬盤(pán)上刪除)。



    3.部署一個(gè)web應(yīng)用

    有兩個(gè)辦法可以在系統(tǒng)中部署web服務(wù)。

    1> 拷貝你的WAR文件或者你的web應(yīng)用文件夾(包括該web的所有內(nèi)容)到$CATALINA_BASE/webapps目錄下。

    2> 為你的web服務(wù)建立一個(gè)只包括context內(nèi)容的XML片斷文件,并把該文件放到$CATALINA_BASE/webapps目錄下。這個(gè)web應(yīng)用本身可以存儲(chǔ)在硬盤(pán)上的任何地方。



    如果你有一個(gè)WAR文件,你若想部署它,則只需要把該文件簡(jiǎn)單的拷貝到CATALINA_BASE/webapps目錄下即可,文件必須以“.war”作為擴(kuò)展名。一旦Tomcat監(jiān)聽(tīng)到這個(gè)文件,它將(缺省的)解開(kāi)該文件包作為一個(gè)子目錄,并以WAR文件的文件名作為子目錄的名字。接下來(lái),Tomcat將在內(nèi)存中建立一個(gè)context,就好象你在server.xml文件里建立一樣。當(dāng)然,其他必需的內(nèi)容,將從server.xml中的DefaultContext獲得。



    部署web應(yīng)用的另一種方式是寫(xiě)一個(gè)Context XML片斷文件,然后把該文件拷貝到CATALINA_BASE/webapps目錄下。一個(gè)Context片斷并非一個(gè)完整的XML文件,而只是一個(gè)context元素,以及對(duì)該應(yīng)用的相應(yīng)描述。這種片斷文件就像是從server.xml中切取出來(lái)的context元素一樣,所以這種片斷被命名為“context片斷”。



    舉個(gè)例子,如果我們想部署一個(gè)名叫MyWebApp.war的應(yīng)用,該應(yīng)用使用realm作為訪問(wèn)控制方式,我們可以使用下面這個(gè)片斷:



    <!--

    Context fragment for deploying MyWebApp.war

    -->

    <Context path="/demo" docBase="webapps/MyWebApp.war"

    debug="0" privileged="true">

    <Realm className="org.apache.catalina.realm.UserDatabaseRealm"

    resourceName="UserDatabase"/>

    </Context>



    把該片斷命名為“MyWebApp.xml”,然后拷貝到CATALINA_BASE/webapps目錄下。



    這種context片斷提供了一種便利的方法來(lái)部署web應(yīng)用,你不需要編輯server.xml,除非你想改變?nèi)笔〉牟渴鹛匦裕惭b一個(gè)新的web應(yīng)用時(shí)不需要重啟動(dòng)Tomcat。





    4.配置虛擬主機(jī)(Virtual Hosts)

    關(guān)于server.xml中“Host”這個(gè)元素,只有在你設(shè)置虛擬主機(jī)的才需要修改。虛擬主機(jī)是一種在一個(gè)web服務(wù)器上服務(wù)多個(gè)域名的機(jī)制,對(duì)每個(gè)域名而言,都好象獨(dú)享了整個(gè)主機(jī)。實(shí)際上,大多數(shù)的小型商務(wù)網(wǎng)站都是采用虛擬主機(jī)實(shí)現(xiàn)的,這主要是因?yàn)樘摂M主機(jī)能直接連接到Internet并提供相應(yīng)的帶寬,以保障合理的訪問(wèn)響應(yīng)速度,另外虛擬主機(jī)還能提供一個(gè)穩(wěn)定的固定IP。



    基于名字的虛擬主機(jī)可以被建立在任何web服務(wù)器上,建立的方法就是通過(guò)在域名服務(wù)器(DNS)上建立IP地址的別名,并且告訴web服務(wù)器把去往不同域名的請(qǐng)求分發(fā)到相應(yīng)的網(wǎng)頁(yè)目錄。因?yàn)檫@篇文章主要是講Tomcat,我們不準(zhǔn)備介紹在各種操作系統(tǒng)上設(shè)置DNS的方法,如果你在這方面需要幫助,請(qǐng)參考《DNS and Bind》一書(shū),作者是Paul Albitz and Cricket Liu (OReilly)。為了示范方便,我將使用一個(gè)靜態(tài)的主機(jī)文件,因?yàn)檫@是測(cè)試別名最簡(jiǎn)單的方法。

    在Tomcat中使用虛擬主機(jī),你需要設(shè)置DNS或主機(jī)數(shù)據(jù)。為了測(cè)試,為本地IP設(shè)置一個(gè)IP別名就足夠了,接下來(lái),你需要在server.xml中添加幾行內(nèi)容,如下:



    <Server port="8005" shutdown="SHUTDOWN" debug="0">

    <Service name="Tomcat-Standalone">

    <Connector className="org.apache.coyote.tomcat4.CoyoteConnector"

    port="8080" minProcessors="5" maxProcessors="75"

    enableLookups="true" redirectPort="8443"/>

    <Connector className="org.apache.coyote.tomcat4.CoyoteConnector"

    port="8443" minProcessors="5" maxProcessors="75"

    acceptCount="10" debug="0" scheme="https" secure="true"/>

    <Factory className="org.apache.coyote.tomcat4.CoyoteServerSocketFactory"

    clientAuth="false" protocol="TLS" />

    </Connector>

    <Engine name="Standalone" defaultHost="localhost" debug="0">

    <!-- This Host is the default Host -->

    <Host name="localhost" debug="0" appBase="webapps"

    unpackWARs="true" autoDeploy="true">

    <Context path="" docBase="ROOT" debug="0"/>

    <Context path="/orders" docBase="/home/ian/orders" debug="0"

    reloadable="true" crossContext="true">

    </Context>

    </Host>



    <!-- This Host is the first "Virtual Host": www.example.com -->

    <Host name="www.example.com" appBase="/home/example/webapp">

    <Context path="" docBase="."/>

    </Host>



    </Engine>

    </Service>

    </Server>



    Tomcat的server.xml文件,在初始狀態(tài)下,只包括一個(gè)虛擬主機(jī),但是它容易被擴(kuò)充到支持多個(gè)虛擬主機(jī)。在前面的例子中展示的是一個(gè)簡(jiǎn)單的server.xml版本,其中粗體部分就是用于添加一個(gè)虛擬主機(jī)。每一個(gè)Host元素必須包括一個(gè)或多個(gè)context元素,所包含的context元素中必須有一個(gè)是默認(rèn)的context,這個(gè)默認(rèn)的context的顯示路徑應(yīng)該為空(例如,path=””)。



    5.配置基礎(chǔ)驗(yàn)證(Basic Authentication)

    容器管理驗(yàn)證方法控制著當(dāng)用戶訪問(wèn)受保護(hù)的web應(yīng)用資源時(shí),如何進(jìn)行用戶的身份鑒別。當(dāng)一個(gè)web應(yīng)用使用了Basic Authentication(BASIC參數(shù)在web.xml文件中auto-method元素中設(shè)置),而有用戶訪問(wèn)受保護(hù)的web應(yīng)用時(shí),Tomcat將通過(guò)HTTP Basic Authentication方式,彈出一個(gè)對(duì)話框,要求用戶輸入用戶名和密碼。在這種驗(yàn)證方法中,所有密碼將被以64位的編碼方式在網(wǎng)絡(luò)上傳輸。



    注意:使用Basic Authentication通過(guò)被認(rèn)為是不安全的,因?yàn)樗鼪](méi)有強(qiáng)健的加密方法,除非在客戶端和服務(wù)器端都使用HTTPS或者其他密碼加密碼方式(比如,在一個(gè)虛擬私人網(wǎng)絡(luò)中)。若沒(méi)有額外的加密方法,網(wǎng)絡(luò)管理員將能夠截獲(或?yàn)E用)用戶的密碼。但是,如果你是剛開(kāi)始使用Tomcat,或者你想在你的web應(yīng)用中測(cè)試一下基于容器的安全管理,Basic Authentication還是非常易于設(shè)置和使用的。只需要添加<security-constraint>和<login-config>兩個(gè)元素到你的web應(yīng)用的web.xml文件中,并且在CATALINA_BASE/conf/tomcat-users.xml 文件中添加適當(dāng)?shù)?lt;role>和<user>即可,然后重新啟動(dòng)Tomcat。



    下面例子中的web.xml摘自一個(gè)俱樂(lè)部會(huì)員網(wǎng)站系統(tǒng),該系統(tǒng)中只有member目錄被保護(hù)起來(lái),并使用Basic Authentication進(jìn)行身份驗(yàn)證。請(qǐng)注意,這種方式將有效的代替Apache web服務(wù)器中的.htaccess文件。



    <!--

    Define the Members-only area, by defining

    a "Security Constraint" on this Application, and

    mapping it to the subdirectory (URL) that we want

    to restrict.

    -->

    <security-constraint>

    <web-resource-collection>

    <web-resource-name>

    Entire Application

    </web-resource-name>

    <url-pattern>/members/*</url-pattern>

    </web-resource-collection>

    <auth-constraint>

    <role-name>member</role-name>

    </auth-constraint>

    </security-constraint>

    <!-- Define the Login Configuration for this Application -->

    <login-config>

    <auth-method>BASIC</auth-method>

    <realm-name>My Club Members-only Area</realm-name>

    </login-config>



    6.配置單點(diǎn)登錄(Single Sign-On)

    一旦你設(shè)置了realm和驗(yàn)證的方法,你就需要進(jìn)行實(shí)際的用戶登錄處理。一般說(shuō)來(lái),對(duì)用戶而言登錄系統(tǒng)是一件很麻煩的事情,你必須盡量減少用戶登錄驗(yàn)證的次數(shù)。作為缺省的情況,當(dāng)用戶第一次請(qǐng)求受保護(hù)的資源時(shí),每一個(gè)web應(yīng)用都會(huì)要求用戶登錄。如果你運(yùn)行了多個(gè)web應(yīng)用,并且每個(gè)應(yīng)用都需要進(jìn)行單獨(dú)的用戶驗(yàn)證,那這看起來(lái)就有點(diǎn)像你在與你的用戶搏斗。用戶們不知道怎樣才能把多個(gè)分離的應(yīng)用整合成一個(gè)單獨(dú)的系統(tǒng),所有他們也就不知道他們需要訪問(wèn)多少個(gè)不同的應(yīng)用,只是很迷惑,為什么總要不停的登錄。



    Tomcat 4的“single sign-on”特性允許用戶在訪問(wèn)同一虛擬主機(jī)下所有web應(yīng)用時(shí),只需登錄一次。為了使用這個(gè)功能,你只需要在Host上添加一個(gè)SingleSignOn Valve元素即可,如下所示:



    <Valve className="org.apache.catalina.authenticator.SingleSignOn"

    debug="0"/>



    在Tomcat初始安裝后,server.xml的注釋里面包括SingleSignOn Valve配置的例子,你只需要去掉注釋,即可使用。那么,任何用戶只要登錄過(guò)一個(gè)應(yīng)用,則對(duì)于同一虛擬主機(jī)下的所有應(yīng)用同樣有效。



    使用single sign-on valve有一些重要的限制:

    1> value必須被配置和嵌套在相同的Host元素里,并且所有需要進(jìn)行單點(diǎn)驗(yàn)證的web應(yīng)用(必須通過(guò)context元素定義)都位于該Host下。

    2> 包括共享用戶信息的realm必須被設(shè)置在同一級(jí)Host中或者嵌套之外。

    3> 不能被context中的realm覆蓋。

    4> 使用單點(diǎn)登錄的web應(yīng)用最好使用一個(gè)Tomcat的內(nèi)置的驗(yàn)證方式(被定義在web.xml中的<auth-method>中),這比自定義的驗(yàn)證方式強(qiáng),Tomcat內(nèi)置的的驗(yàn)證方式包括basic、digest、form和client-cert。

    5> 如果你使用單點(diǎn)登錄,還希望集成一個(gè)第三方的web應(yīng)用到你的網(wǎng)站中來(lái),并且這個(gè)新的web應(yīng)用使用它自己的驗(yàn)證方式,而不使用容器管理安全,那你基本上就沒(méi)招了。你的用戶每次登錄原來(lái)所有應(yīng)用時(shí)需要登錄一次,并且在請(qǐng)求新的第三方應(yīng)用時(shí)還得再登錄一次。當(dāng)然,如果你擁有這個(gè)第三方web應(yīng)用的源碼,而你又是一個(gè)程序員,你可以修改它,但那恐怕也不容易做。

    6> 單點(diǎn)登錄需要使用cookies。



    7.配置用戶定制目錄(Customized User Directores)

    一些站點(diǎn)允許個(gè)別用戶在服務(wù)器上發(fā)布網(wǎng)頁(yè)。例如,一所大學(xué)的學(xué)院可能想給每一位學(xué)生一個(gè)公共區(qū)域,或者是一個(gè)ISP希望給一些web空間給他的客戶,但這又不是虛擬主機(jī)。在這種情況下,一個(gè)典型的方法就是在用戶名前面加一個(gè)特殊字符(~),作為每位用戶的網(wǎng)站,比如:



    http://www.cs.myuniversity.edu/~username

    http://members.mybigisp.com/~username



    Tomcat提供兩種方法在主機(jī)上映射這些個(gè)人網(wǎng)站,主要使用一對(duì)特殊的Listener元素。Listener的className屬性應(yīng)該是org.apache.catalina.startup.UserConfig,userClass屬性應(yīng)該是幾個(gè)映射類之一。如果你的系統(tǒng)是Unix,它將有一個(gè)標(biāo)準(zhǔn)的/etc/passwd文件,該文件中的帳號(hào)能夠被運(yùn)行中的Tomcat很容易的讀取,該文件指定了用戶的主目錄,使用PasswdUserDatabase 映射類。



    <Listener className="org.apache.catalina.startup.UserConfig"

    directoryName="public_html"

    userClass="org.apache.catalina.startup.PasswdUserDatabase"/>



    web文件需要放置在像/home/users/ian/public_html 或者 /users/jbrittain/public_html一樣的目錄下面。當(dāng)然你也可以改變public_html 到其他任何子目錄下。



    實(shí)際上,這個(gè)用戶目錄根本不一定需要位于用戶主目錄下里面。如果你沒(méi)有一個(gè)密碼文件,但你又想把一個(gè)用戶名映射到公共的像/home一樣目錄的子目錄里面,則可以使用HomesUserDatabase類。



    <Listener className="org.apache.catalina.startup.UserConfig"

    directoryName="public_html" homeBase="/home"

    userClass="org.apache.catalina.startup.HomesUserDatabase"/>



    這樣一來(lái),web文件就可以位于像/home/ian/public_html 或者 /home/jasonb/public_html一樣的目錄下。這種形式對(duì)Windows而言更加有利,你可以使用一個(gè)像c:home這樣的目錄。



    這些Listener元素,如果出現(xiàn),則必須在Host元素里面,而不能在context元素里面,因?yàn)樗鼈兌加脩?yīng)用于Host本身。





    8.在Tomcat中使用CGI腳本

    Tomcat主要是作為Servlet/JSP容器,但它也有許多傳統(tǒng)web服務(wù)器的性能。支持通用網(wǎng)關(guān)接口(Common Gateway Interface,即CGI)就是其中之一,CGI提供一組方法在響應(yīng)瀏覽器請(qǐng)求時(shí)運(yùn)行一些擴(kuò)展程序。CGI之所以被稱為通用,是因?yàn)樗茉诖蠖鄶?shù)程序或腳本中被調(diào)用,包括:Perl,Python,awk,Unix shell scripting等,甚至包括Java。當(dāng)然,你大概不會(huì)把一個(gè)Java應(yīng)用程序當(dāng)作CGI來(lái)運(yùn)行,畢竟這樣太過(guò)原始。一般而言,開(kāi)發(fā)Servlet總要比CGI具有更好的效率,因?yàn)楫?dāng)用戶點(diǎn)擊一個(gè)鏈接或一個(gè)按鈕時(shí),你不需要從操作系統(tǒng)層開(kāi)始進(jìn)行處理。



    Tomcat包括一個(gè)可選的CGI Servlet,允許你運(yùn)行遺留下來(lái)的CGI腳本。



    為了使Tomcat能夠運(yùn)行CGI,你必須做如下幾件事:

    1. 把servlets-cgi.renametojar (在CATALINA_HOME/server/lib/目錄下)改名為servlets-cgi.jar。處理CGI的servlet應(yīng)該位于Tomcat的CLASSPATH下。

    2. 在Tomcat的CATALINA_BASE/conf/web.xml 文件中,把關(guān)于<servlet-name> CGI的那段的注釋去掉(默認(rèn)情況下,該段位于第241行)。

    3. 同樣,在Tomcat的CATALINA_BASE/conf/web.xml文件中,把關(guān)于對(duì)CGI進(jìn)行映射的那段的注釋去掉(默認(rèn)情況下,該段位于第299行)。注意,這段內(nèi)容指定了HTML鏈接到CGI腳本的訪問(wèn)方式。

    4. 你可以把CGI腳本放置在WEB-INF/cgi 目錄下(注意,WEB-INF是一個(gè)安全的地方,你可以把一些不想被用戶看見(jiàn)或基于安全考慮不想暴露的文件放在此處),或者你也可以把CGI腳本放置在context下的其他目錄下,并為CGI Servlet調(diào)整cgiPathPrefix初始化參數(shù)。這就指定的CGI Servlet的實(shí)際位置,且不能與上一步指定的URL重名。

    5. 重新啟動(dòng)Tomcat,你的CGI就可以運(yùn)行了。



    在Tomcat中,CGI程序缺省放置在WEB-INF/cgi目錄下,正如前面所提示的那樣,WEB-INF目錄受保護(hù)的,通過(guò)客戶端的瀏覽器無(wú)法窺探到其中內(nèi)容,所以對(duì)于放置含有密碼或其他敏感信息的CGI腳本而言,這是一個(gè)非常好的地方。為了兼容其他服務(wù)器,盡管你也可以把CGI腳本保存在傳統(tǒng)的/cgi-bin目錄,但要知道,在這些目錄中的文件有可能被網(wǎng)上好奇的沖浪者看到。另外,在Unix中,請(qǐng)確定運(yùn)行Tomcat的用戶有執(zhí)行CGI腳本的權(quán)限。



    9.改變Tomcat中的JSP編譯器(JSP Compiler)

    在Tomcat 4.1(或更高版本,大概),JSP的編譯由包含在Tomcat里面的Ant程序控制器直接執(zhí)行。這聽(tīng)起來(lái)有一點(diǎn)點(diǎn)奇怪,但這正是Ant有意為之的一部分,有一個(gè)API文檔指導(dǎo)開(kāi)發(fā)者在沒(méi)有啟動(dòng)一個(gè)新的JVM的情況下,使用Ant。這是使用Ant進(jìn)行Java開(kāi)發(fā)的一大優(yōu)勢(shì)。另外,這也意味著你現(xiàn)在能夠在Ant中使用任何javac支持的編譯方式,這里有一個(gè)關(guān)于Apache Ant使用手冊(cè)的javac page列表。使用起來(lái)是容易的,因?yàn)槟阒恍枰?lt;init-param> 元素中定義一個(gè)名字叫“compiler”,并且在value中有一個(gè)支持編譯的編譯器名字,示例如下:



    <servlet>

    <servlet-name>jsp</servlet-name>

    <servlet-class>

    org.apache.jasper.servlet.JspServlet

    </servlet-class>

    <init-param>

    <param-name>logVerbosityLevel</param-name>

    <param-value>WARNING</param-value>

    </init-param>

    <init-param>

    <param-name>compiler</param-name>

    <param-value>jikes</param-value>

    </init-param>

    <load-on-startup>3</load-on-startup>

    </servlet>



    當(dāng)然,給出的編譯器必須已經(jīng)安裝在你的系統(tǒng)中,并且CLASSPATH可能需要設(shè)置,那處決于你選擇的是何種編譯器。



    10.限制特定主機(jī)訪問(wèn)(Restricting Access to Specific Hosts)

    有時(shí),你可能想限制對(duì)Tomcat web應(yīng)用的訪問(wèn),比如,你希望只有你指定的主機(jī)或IP地址可以訪問(wèn)你的應(yīng)用。這樣一來(lái),就只有那些指定的的客戶端可以訪問(wèn)服務(wù)的內(nèi)容了。為了實(shí)現(xiàn)這種效果,Tomcat提供了兩個(gè)參數(shù)供你配置:RemoteHostValve 和RemoteAddrValve。



    通過(guò)配置這兩個(gè)參數(shù),可以讓你過(guò)濾來(lái)自請(qǐng)求的主機(jī)或IP地址,并允許或拒絕哪些主機(jī)/IP。與之類似的,在Apache的httpd文件里有對(duì)每個(gè)目錄的允許/拒絕指定。

    例如你可以把Admin Web application設(shè)置成只允許本地訪問(wèn),設(shè)置如下:



    <Context path="/path/to/secret_files" ...>

    <Valve className="org.apache.catalina.valves.RemoteAddrValve"

    allow="127.0.0.1" deny=""/>

    </Context>



    如果沒(méi)有給出允許主機(jī)的指定,那么與拒絕主機(jī)匹配的主機(jī)就會(huì)被拒絕,除此之外的都是允許的。與之類似,如果沒(méi)有給出拒絕主機(jī)的指定,那么與允許主機(jī)匹配的主機(jī)就會(huì)被允許,除此之外的都是拒絕的。



    --------------------------------------



    作者簡(jiǎn)介:

    Jason Brittain是CollabNet公司的一名資深軟件工程師,主要負(fù)責(zé)軟件底層架構(gòu)的開(kāi)發(fā)。他已經(jīng)為Apache Jakarta項(xiàng)目做了很多貢獻(xiàn),多年以來(lái),他一直是一名積極的開(kāi)源軟件開(kāi)發(fā)者。



    Ian F. Darwin已經(jīng)在計(jì)算機(jī)行業(yè)工作了30年:從1980年開(kāi)始使用Unix,從1995年開(kāi)始使用Java,從1998年開(kāi)始使用OpenBSD。他是兩本Oreilly圖書(shū)的作者:Checking C Programs with lint 和 Java Cookbook,還與Jason Brittain合著了Tomcat: The Definitive Guide。



    譯者簡(jiǎn)介:

    陳光(Holen Chen),J2EE項(xiàng)目經(jīng)理,熟悉知識(shí)管理及電子政務(wù),致力于Apache Jakarta項(xiàng)目在國(guó)廣的推廣及深層次應(yīng)用,可通過(guò) holen@263.net與作者聯(lián)系。

    posted @ 2006-05-15 10:00 javaGrowing 閱讀(276) | 評(píng)論 (0)編輯 收藏

    下面給出得Log4J配置文件實(shí)現(xiàn)了輸出到控制臺(tái),文件,回滾文件,發(fā)送日志郵件,輸出到數(shù)據(jù)庫(kù)日志表,自定義標(biāo)簽等全套功能。

    log4j.rootLogger=DEBUG,CONSOLE,A1,im
    #DEBUG,CONSOLE,FILE,ROLLING_FILE,MAIL,DATABASE

    log4j.addivity.org.apache=true


    ###################
    # Console Appender
    ###################
    log4j.appender.CONSOLE=org.apache.log4j.ConsoleAppender
    log4j.appender.Threshold=DEBUG
    log4j.appender.CONSOLE.Target=System.out
    log4j.appender.CONSOLE.layout=org.apache.log4j.PatternLayout
    log4j.appender.CONSOLE.layout.ConversionPattern=[framework] %d - %c -%-4r [%t] %-5p %c %x - %m%n
    #log4j.appender.CONSOLE.layout.ConversionPattern=[start]%d{DATE}[DATE]%n%p[PRIORITY]%n%x[NDC]%n%t[THREAD] n%c[CATEGORY]%n%m[MESSAGE]%n%n


    #####################
    # File Appender
    #####################
    log4j.appender.FILE=org.apache.log4j.FileAppender
    log4j.appender.FILE.File=file.log
    log4j.appender.FILE.Append=false
    log4j.appender.FILE.layout=org.apache.log4j.PatternLayout
    log4j.appender.FILE.layout.ConversionPattern=[framework] %d - %c -%-4r [%t] %-5p %c %x - %m%n
    # Use this layout for LogFactor 5 analysis



    ########################
    # Rolling File
    ########################
    log4j.appender.ROLLING_FILE=org.apache.log4j.RollingFileAppender
    log4j.appender.ROLLING_FILE.Threshold=ERROR
    log4j.appender.ROLLING_FILE.File=rolling.log
    log4j.appender.ROLLING_FILE.Append=true
    log4j.appender.ROLLING_FILE.MaxFileSize=10KB
    log4j.appender.ROLLING_FILE.MaxBackupIndex=1
    log4j.appender.ROLLING_FILE.layout=org.apache.log4j.PatternLayout
    log4j.appender.ROLLING_FILE.layout.ConversionPattern=[framework] %d - %c -%-4r [%t] %-5p %c %x - %m%n


    ####################
    # Socket Appender
    ####################
    log4j.appender.SOCKET=org.apache.log4j.RollingFileAppender
    log4j.appender.SOCKET.RemoteHost=localhost
    log4j.appender.SOCKET.Port=5001
    log4j.appender.SOCKET.LocationInfo=true
    # Set up for Log Facter 5
    log4j.appender.SOCKET.layout=org.apache.log4j.PatternLayout
    log4j.appender.SOCET.layout.ConversionPattern=[start]%d{DATE}[DATE]%n%p[PRIORITY]%n%x[NDC]%n%t[THREAD]%n%c[CATEGORY]%n%m[MESSAGE]%n%n


    ########################
    # Log Factor 5 Appender
    ########################
    log4j.appender.LF5_APPENDER=org.apache.log4j.lf5.LF5Appender
    log4j.appender.LF5_APPENDER.MaxNumberOfRecords=2000


    ########################
    # SMTP Appender
    #######################
    log4j.appender.MAIL=org.apache.log4j.net.SMTPAppender
    log4j.appender.MAIL.Threshold=FATAL
    log4j.appender.MAIL.BufferSize=10
    log4j.appender.MAIL.From=chenyl@hollycrm.com
    log4j.appender.MAIL.SMTPHost=mail.hollycrm.com
    log4j.appender.MAIL.Subject=Log4J Message
    log4j.appender.MAIL.To=chenyl@hollycrm.com
    log4j.appender.MAIL.layout=org.apache.log4j.PatternLayout
    log4j.appender.MAIL.layout.ConversionPattern=[framework] %d - %c -%-4r [%t] %-5p %c %x - %m%n


    ########################
    # JDBC Appender
    #######################
    log4j.appender.DATABASE=org.apache.log4j.jdbc.JDBCAppender
    log4j.appender.DATABASE.URL=jdbc:mysql://localhost:3306/test
    log4j.appender.DATABASE.driver=com.mysql.jdbc.Driver
    log4j.appender.DATABASE.user=root
    log4j.appender.DATABASE.password=
    log4j.appender.DATABASE.sql=INSERT INTO LOG4J (Message) VALUES ('[framework] %d - %c -%-4r [%t] %-5p %c %x - %m%n')
    log4j.appender.DATABASE.layout=org.apache.log4j.PatternLayout
    log4j.appender.DATABASE.layout.ConversionPattern=[framework] %d - %c -%-4r [%t] %-5p %c %x - %m%n


    log4j.appender.A1=org.apache.log4j.DailyRollingFileAppender
    log4j.appender.A1.File=SampleMessages.log4j
    log4j.appender.A1.DatePattern=yyyyMMdd-HH'.log4j'
    log4j.appender.A1.layout=org.apache.log4j.xml.XMLLayout

    ###################
    #自定義Appender
    ###################
    log4j.appender.im = net.cybercorlin.util.logger.appender.IMAppender

    log4j.appender.im.host = mail.cybercorlin.net
    log4j.appender.im.username = username
    log4j.appender.im.password = password
    log4j.appender.im.recipient = corlin@cybercorlin.net

    log4j.appender.im.layout=org.apache.log4j.PatternLayout
    log4j.appender.im.layout.ConversionPattern =[framework] %d - %c -%-4r [%t] %-5p %c %x - %m%n

    posted @ 2006-05-15 09:57 javaGrowing 閱讀(248) | 評(píng)論 (0)編輯 收藏

    log4j.rootLogger=DEBUG,CONSOLE,A1,im
    log4j.addivity.org.apache=true

    ?

    #?應(yīng)用于控制臺(tái)

    log4j.appender.CONSOLE=org.apache.log4j.ConsoleAppender
    log4j.appender.Threshold=DEBUG
    log4j.appender.CONSOLE.Target=System.out
    log4j.appender.CONSOLE.layout=org.apache.log4j.PatternLayout
    log4j.appender.CONSOLE.layout.ConversionPattern=[framework]?%d?-?%c?-%-4r?[%t]?%-5p?%c?%x?-?%m%n
    #log4j.appender.CONSOLE.layout.ConversionPattern=[start]%d{DATE}[DATE]%n%p[PRIORITY]%n%x[NDC]%n%t[THREAD]?n%c[CATEGORY]%n%m[MESSAGE]%n%n


    #應(yīng)用于文件

    log4j.appender.FILE=org.apache.log4j.FileAppender
    log4j.appender.FILE.File=file.log
    log4j.appender.FILE.Append=false
    log4j.appender.FILE.layout=org.apache.log4j.PatternLayout
    log4j.appender.FILE.layout.ConversionPattern=[framework]?%d?-?%c?-%-4r?[%t]?%-5p?%c?%x?-?%m%n
    #?Use?this?layout?for?LogFactor?5?analysis



    #?應(yīng)用于文件回滾

    log4j.appender.ROLLING_FILE=org.apache.log4j.RollingFileAppender
    log4j.appender.ROLLING_FILE.Threshold=ERROR
    log4j.appender.ROLLING_FILE.File=rolling.log
    log4j.appender.ROLLING_FILE.Append=true
    log4j.appender.ROLLING_FILE.MaxFileSize=10KB
    log4j.appender.ROLLING_FILE.MaxBackupIndex=1
    log4j.appender.ROLLING_FILE.layout=org.apache.log4j.PatternLayout
    log4j.appender.ROLLING_FILE.layout.ConversionPattern=[framework]?%d?-?%c?-%-4r?[%t]?%-5p?%c?%x?-?%m%n


    #應(yīng)用于socket
    log4j.appender.SOCKET=org.apache.log4j.RollingFileAppender
    log4j.appender.SOCKET.RemoteHost=localhost
    log4j.appender.SOCKET.Port=5001
    log4j.appender.SOCKET.LocationInfo=true
    #?Set?up?for?Log?Facter?5
    log4j.appender.SOCKET.layout=org.apache.log4j.PatternLayout
    log4j.appender.SOCET.layout.ConversionPattern=[start]%d{DATE}[DATE]%n%p[PRIORITY]%n%x[NDC]%n%t[THREAD]%n%c[CATEGORY]%n%m[MESSAGE]%n%n


    #?Log?Factor?5?Appender
    log4j.appender.LF5_APPENDER=org.apache.log4j.lf5.LF5Appender
    log4j.appender.LF5_APPENDER.MaxNumberOfRecords=2000



    #?發(fā)送日志給郵件

    log4j.appender.MAIL=org.apache.log4j.net.SMTPAppender
    log4j.appender.MAIL.Threshold=FATAL
    log4j.appender.MAIL.BufferSize=10
    log4j.appender.MAIL.From=web@www.wuset.com
    log4j.appender.MAIL.SMTPHost=www.wusetu.com
    log4j.appender.MAIL.Subject=Log4J?Message
    log4j.appender.MAIL.To=web@www.wusetu.com
    log4j.appender.MAIL.layout=org.apache.log4j.PatternLayout
    log4j.appender.MAIL.layout.ConversionPattern=[framework]?%d?-?%c?-%-4r?[%t]?%-5p?%c?%x?-?%m%n



    #?用于數(shù)據(jù)庫(kù)
    log4j.appender.DATABASE=org.apache.log4j.jdbc.JDBCAppender
    log4j.appender.DATABASE.URL=jdbc:mysql://localhost:3306/test
    log4j.appender.DATABASE.driver=com.mysql.jdbc.Driver
    log4j.appender.DATABASE.user=root
    log4j.appender.DATABASE.password=
    log4j.appender.DATABASE.sql=INSERT?INTO?LOG4J?(Message)?VALUES?('[framework]?%d?-?%c?-%-4r?[%t]?%-5p?%c?%x?-?%m%n')
    log4j.appender.DATABASE.layout=org.apache.log4j.PatternLayout
    log4j.appender.DATABASE.layout.ConversionPattern=[framework]?%d?-?%c?-%-4r?[%t]?%-5p?%c?%x?-?%m%n


    log4j.appender.A1=org.apache.log4j.DailyRollingFileAppender
    log4j.appender.A1.File=SampleMessages.log4j
    log4j.appender.A1.DatePattern=yyyyMMdd-HH'.log4j'
    log4j.appender.A1.layout=org.apache.log4j.xml.XMLLayout

    #自定義Appender

    log4j.appender.im?=?net.cybercorlin.util.logger.appender.IMAppender

    log4j.appender.im.host?=?mail.cybercorlin.net
    log4j.appender.im.username?=?username
    log4j.appender.im.password?=?password
    log4j.appender.im.recipient?=?corlin@cybercorlin.net

    log4j.appender.im.layout=org.apache.log4j.PatternLayout
    log4j.appender.im.layout.ConversionPattern?=[framework]?%d?-?%c?-%-4r?[%t]?%-5p?%c?%x?-?%m%n

    posted @ 2006-05-15 09:35 javaGrowing 閱讀(352) | 評(píng)論 (0)編輯 收藏

    僅列出標(biāo)題
    共19頁(yè): First 上一頁(yè) 7 8 9 10 11 12 13 14 15 下一頁(yè) Last 
    主站蜘蛛池模板: 亚洲一区二区三区无码国产| 黄色视频在线免费观看| 无码人妻久久一区二区三区免费| 天堂在线免费观看中文版| 亚洲福利在线观看| 一区二区三区在线观看免费| 成年在线观看免费人视频草莓| 亚洲AV人无码综合在线观看| 一本一道dvd在线观看免费视频| 毛片免费在线观看网站| 亚洲综合激情另类小说区| 在线观看免费视频网站色| 免费国产不卡午夜福在线| 国产精品亚洲一区二区麻豆| 999任你躁在线精品免费不卡| 国产亚洲成归v人片在线观看| 亚洲Aⅴ在线无码播放毛片一线天| 99视频全部免费精品全部四虎| 久久精品国产亚洲AV麻豆~| 一级毛片免费不卡直观看| 国产一级高清免费观看| 亚洲色无码国产精品网站可下载| 免费A级毛片无码A∨免费| 亚洲AV无码国产精品麻豆天美| 久久久久久久久久久免费精品| 亚洲?V无码乱码国产精品| 亚洲AV无码一区二区三区网址 | 亚洲精品免费在线观看| 精品国产亚洲男女在线线电影 | 国产一级高清免费观看| 亚洲精品成a人在线观看☆| 99在线视频免费观看视频| 亚洲精品欧洲精品| 91免费国产精品| 久久亚洲国产成人精品性色| 久久久久免费看黄a级试看| 亚洲av中文无码乱人伦在线r▽| a级片免费在线播放| 亚洲精品夜夜夜妓女网| a级黄色毛片免费播放视频| 亚洲熟女一区二区三区|