javaGrowing

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

          #

          正則表達式:

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

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

          由于本文目的是要介紹如何在JAVA里運用正則表達式,因此對剛接觸正則表達式的讀者請參考有關資料,在此因篇幅有限不作介紹。





          回頁首


          JAVA對正則表達式的支持:

          在JDK1.3或之前的JDK版本中并沒有包含正則表達式庫可供JAVA程序員使用,之前我們一般都在使用第三方提供的正則表達式庫,這些第三方庫中有源代碼開放的,也有需付費購買的,而現時在JDK1.4的測試版中也已經包含有正則表達式庫---java.util.regex。

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





          回頁首


          Jakarta-ORO正則表達式庫

          1.簡介:

          Jakarta-ORO是最全面以及優化得最好的正則表達式API之一,Jakarta-ORO庫以前叫做OROMatcher,是由Daniel F. Savarese編寫,后來他將其贈與Jakarta Project,讀者可在Apache.org的網站 下載 該API包。

          許多源代碼開放的正則表達式庫都是支持Perl5兼容的正則表達式語法,Jakarta-ORO正則表達式庫也不例外,他與Perl 5正則表達式完全兼容。

          2.對象與其方法:

          ★PatternCompiler對象:
          我們在使用Jakarta-ORO API包時,最先要做的是,創建一個Perl5Compiler類的實例,并把它賦值給PatternCompiler接口對象。Perl5Compiler是PatternCompiler接口的一個實現,允許你把正則表達式編譯成用來匹配的Pattern對象。

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

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

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

          在默認的情況下,編譯器會創建一個對大小寫敏感的模式(pattern)。因此,上面代碼編譯得到的模式只匹配"say"、"shy"、 "sky"和"sly",但不匹配"Say"和"skY"。要創建一個大小寫不敏感的模式,你應該在調用編譯器的時候指定一個額外的參數:
          pattern=compiler.compile("s[ahkl]y",Perl5Compiler.CASE_INSENSITIVE_MASK);

          Pattern對象創建好之后,就可以通過PatternMatcher類用該Pattern對象進行模式匹配。

          ★PatternMatcher對象:

          PatternMatcher對象依據Pattern對象和字符串展開匹配檢查。你要實例化一個Perl5Matcher類并把結果賦值給PatternMatcher接口。Perl5Matcher類是PatternMatcher接口的一個實現,它根據Perl 5正則表達式語法進行模式匹配:
          PatternMatcher matcher=new Perl5Matcher();

          PatternMatcher對象提供了多個方法進行匹配操作,這些方法的第一個參數都是需要根據正則表達式進行匹配的字符串:

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

          但以上三種方法只會查找輸入字符串中匹配正則表達式的第一個對象,如果當字符串可能有多個子串匹配給定的正則表達式時,那么你就可以在調用上面三個方法時用PatternMatcherInput對象作為參數替代String對象,這樣就可以從字符串中最后一次匹配的位置開始繼續進行匹配,這樣就方便的多了。

          用PatternMatcherInput對象作為參數替代String時,上述三個方法的語法如下:

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

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

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

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

          在這里我相信有必要詳細解說一下第三個參數Substiution對象,因為它將決定替換將怎樣進行。

          Substiution:
          Substiution是一個接口類,它為你提供了在使用Util.substitute()方法時控制替換方式的手段,它有兩個標準的實現類:StringSubstitution與Perl5Substitution。當然,同時你也可以生成自己的實現類來定制你所需要的特殊替換動作。

          StringSubstitution:
          StringSubstitution 實現的是簡單的純文字替換手段,它有兩個構造方法:

          StringSubstitution()->缺省的構造方法,初始化一個包含零長度字符串的替換對象。

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

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

          它有三個構造器:

          Perl5Substitution()

          Perl5Substitution(java.lang.String substitution)

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

          前兩種構造方法與StringSubstitution一樣,而第三種構造方法下面將會介紹到。

          Perl5Substitution的替換字符串中可以包含用來替代在正則表達式里由小擴號圍起來的匹配組的變量,這些變量是由$1, $2,$3等形式來標識。我們可以用一個例子來解釋怎樣使用替換變量來進行替換:

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

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

          中的numInterpolations參數設為INTERPOLATE_ALL,那么當每次找到一個匹配字串時,替換變量($1,$2等)所指向的內容都根據目前匹配字串來更新,但是如果numInterpolations參數設為一個正整數N時,那么在替換時就只會在前N次匹配發生時替換變量會跟隨匹配對象來調整所代表的內容,但N次之后就以一致以第N次替換變量所代表內容來做為以后替換結果。

          舉個例子會更好理解:

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

          但是如果你把numInterpolations設為2,并且numSubs依然設為SUBSTITUTE_ALL,那么這時你獲得的結果則會是: Tank a123- 85 Tank a256- 32 Tank a256- 22

          你要注意到最后一個替換所用變量$1所代表的內容與第二個$1一樣為"256",而不是預期的"78",因為在替換進行中,替換變量$1只根據匹配內容進行了兩次更新,最后一次就使第二次匹配時所更新的結果,那么我們可以由此知道,如果numInterpolations設為1,那么結果將是: Tank a123- 85 Tank a123- 32 Tank a123- 22

          3.應用示例:

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

          表名:AESOP, 表中每條記錄包含有三列:
          ID(int):單詞索引號
          WORD(varchar):單詞
          CONTENT(varchar):存放單詞的相關解釋與例句等內容

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

          如對應單詞Kevin,CONTENT中的內容如下:
          ['kevin] [名詞](人名凱文){(Kevin loves comic./凱文愛漫畫/名詞: 凱文)( Kevin is living in ZhuHai now./凱文現住在珠海/名詞: 凱文)}

          我們的例子主要針對CONTENT列中內容進行字符串處理。

          ★查找單個匹配:
          首先,讓我們嘗試把CONTNET列中的[音標]字段的內容列示出來,由于所有單詞的記錄中都有這一項并且都在字串開始位置,所以這個查找工作比較簡單:

          1. 確定相應的正則表達式:\[[^]]+\]

            這個是很簡單的正則表達式,其意思是要求相匹配的字符串必須為以一對中括號包含的所有內容,如['kevin] 、[名詞]等,但內容中不包括"]"符號,也就是要避免出現"[][]"會作為一個匹配對象的情況出現(有關正則表達式的基礎知識請參照有關資料,這里不再詳述)。

            注意,在Java中,你必須對每一個向前的斜杠("\")進行轉義處理。所以我們要在上面的正則表達式里每個"\"前面加上一個"\"以免出現編譯錯誤,也就是在JAVA中初始化正則表達式的字符串的語句應該為:

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

            并且在表達式里每個符號中間不能有空格,否則就會同樣出現編譯錯誤。

          2. 實例化PatternCompiler對象,創建Pattern對象

            PatternCompiler compiler=new Perl5Compiler();

            Pattern pattern=compiler.compile(restring);

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

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

            具體實現的完整的程序代碼如下:

            																
            																
            																
            																																						package RegularExpressions;
            //import……
            import org.apache.oro.text.regex.*;
            //使用Jakarta-ORO正則表達式庫前需要把它加到CLASSPATH里面,如果用IDE是//JBUILDER,那么也可以在JBUILDER里直接自建新庫。
            
            public class yisuo{
              public static void main(String[] args){
              try{     
            //使用JDBC DRIVER進行DBMS連接,這里我使用的是一個第三方JDBC 
            //DRIVER,Microsoft本身也有一個面向SQLSERVER7/2000的免費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庫而創建相應的對象
            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+"的音標為:"+pure);
                        }
                      }
                   }
              catch(Exception e) {
                         System.out.println(e);
                   }
              }
            }
            																		
            																
            																
            																
            														

            輸出結果為:kevin的音標為['kevin]

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

          用程序驗證,輸出結果同樣為:kevin的音標為['kevin]

          而如果正則表達式為(\[[^]]+\])(\[[^]]+\]),則會查找到兩個連續的方括號所包含的內容,也就找到[音標] [詞性]兩項,但是兩項的結果分別在兩個組里面,分別由下面語句獲得結果:

          result.group(0)->返回[音標] [詞性]兩項內容,也就是與整個正則表達式相匹配的結果字符串,在這里也就為['kevin] [名詞]

          result.group(1) ->返回[音標]項內容,結果應是['kevin]

          result.group(2) ->返回[詞性]項內容,結果應是[名詞]

          繼續用程序驗證,發現輸出并不正確,主要是當內容有中文時就不能成功匹配,考慮到可能是Jakarta-ORO正則表達式庫版本不支持中文的問題,回看一下原來我一直用的還是2.0.1的老版本,馬上到Jakarta.org上下載最新的2.0.4版本裝上再用程序驗證,得出的結果就和預期一樣正確。

          ★查找多個匹配:
          經過第一步的嘗試使用Jakarta-ORO后,我們已經知道了如何正確使用該API包來查找目標字符串里一個匹配的子串,下面我們接著來看一看當目標字符串里包含不止一個匹配的子串時我們如何把它們一個接一個找出來進行相應的處理。

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

          我們已經知道在匹配操作的三個方法里只要用PatternMatcherInput對象作為參數替代String對象就可以從字符串中最后一次匹配的位置開始繼續進行匹配,實現的程序片段如下:

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

          輸出結果為:['kevin]
          [名詞]

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

          第一步當然是要定出相應的正則表達式,需要有兩個,一是和整個例句部分(也就是由大括號包起來的部分)匹配的正則表達式:"\{.+\}",

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


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

          為了簡便起見,我們不再和從數據庫里讀出,而是構造一個包含同樣內容的字符串變量,程序片段如下:

          												
          														try{
                   String content="['kevin] [名詞](人名凱文){(Kevin loves comic./凱文愛漫畫/名詞:凱文) (Kevin is living in ZhuHai now./凱文現住在珠海/名詞: 凱文)}";
                   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();
          //先找出整個例句部分
                      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();
                  //把每個例句里的各項用分組的辦法分隔出來
                          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);
                 }
          
          												
          										

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

          ★查找替換:
          以上的兩個應用都是單純在查找字符串匹配方面的,我們再來看一下查找后如何對目標字符串進行替換。

          例如我現在想把第二個例句進行改動,換為:Kevin has seen《LEON》seveal times,because it is a good film./ 凱文已經看過《這個殺手不太冷》幾次了,因為它是一部好電影。/名詞:凱文。

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

          改為:
          ['kevin] [名詞](人名凱文){(Kevin loves comic./凱文愛漫畫/名詞: 凱文)( Kevin has seen《LEON》seveal times,because it is a good film./ 凱文已經看過《這個殺手不太冷》幾次了,因為它是一部好電影。/名詞:凱文。)}

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

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

          根據上面的正則表達式Perl5Substitution類可以這樣構造: Perl5Substitution("{$1( Kevin has seen《LEON》seveal times,because it is a good film./ 凱文已經看過《這個殺手不太冷》幾次了,因為它是一部好電影。/名詞:凱文。)}")

          再根據這個Perl5Substitution對象來使用Util.substitute()方法便可以完成替換了,實現的代碼片段如下:

          												
          														try{
             String content="['kevin] [名詞](人名凱文){(Kevin loves comic.
             /凱文愛漫畫/名詞: 凱文)(Kevin lives in ZhuHai now./凱文現住在珠海/名詞: 凱文)}";
             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./ 
                 凱文已經看過《這個殺手不太冷》幾次了,因為它是一部好電影。/名詞:凱文。)}",1),
                  content,Util.SUBSTITUTE_ALL);
                  System.out.println(result);
             }
            catch(Exception e) {
                       System.out.println(e);
                 }
          
          												
          										

          輸出結果是正確的,為:
          ['kevin] [名詞](人名凱文){(Kevin loves comic./凱文愛漫畫/名詞: 凱文)( Kevin has seen《LEON》seveal times,because it is a good film./ 凱文已經看過《這個殺手不太冷》幾次了,因為它是一部好電影。/名詞:凱文。)}

          至于有關使用numInterpolations參數的構造器用法,讀者只要根據上面的介紹自己動手試一下就會清楚了,在此就不再例述。





          回頁首


          總結:

          本文首先介紹了Jakarta-ORO正則表達式庫的對象與方法,并且接著舉例讓讀者對實際應用有進一步的了解,雖然例子都比較簡單,但希望讀者們在看了該文后對Jakarta-ORO正則表達式庫有一定的認知,在實際工作中有所幫助與啟發。

          其實在Jakarta org里除了Jakarta-ORO外還有一個百分百的純JAVA正則表達式庫,就是由Jonathan Locke贈與Jakarta ORG的Regexp,在該包里面包含了完整的文檔以及一個用于調試的Applet例子,對其有興趣的讀者可以到此 下載

          posted @ 2006-05-22 13:19 javaGrowing 閱讀(1226) | 評論 (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 閱讀(734) | 評論 (1)編輯 收藏

          編者按:現在開發Java Web應用,建立和部署Web內容是一件很簡單的工作。使用Jakarta Tomcat作為Servlet和JSP容器的人已經遍及全世界。Tomcat具有免費、跨平臺等諸多特性,并且更新得很快,現在非常的流行。



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



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

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



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

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



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



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



          <role name="admin"/>



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



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



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



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

          Manager Web Application讓你通過一個比Admin Web Application更為簡單的用戶界面,執行一些簡單的Web應用任務。



          Manager Web Application被被定義在一個自動部署文件中:



          CATALINA_BASE/webapps/manager.xml 。



          你必須編輯這個文件,以確保context的docBase參數是絕對路徑,也就是說CATALINA_HOME/server/webapps/manager的絕對路徑。

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



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



          <role name=”manager”>



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



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



          然后重新啟動Tomcat,訪問http://localhost/manager/list,將看到一個很樸素的文本型管理界面,或者訪問http://localhost/manager/html/list,將看到一個HMTL的管理界面。不管是哪種方式都說明你的Manager Web Application現在已經啟動了。



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



          Manager application還允許你停止、重新啟動、移除以及重新部署一個web應用。停止一個應用使其無法被訪問,當有用戶嘗試訪問這個被停止的應用時,將看到一個503的錯誤——“503 - This application is not currently available”。



          移除一個web應用,只是指從Tomcat的運行拷貝中刪除了該應用,如果你重新啟動Tomcat,被刪除的應用將再次出現(也就是說,移除并不是指從硬盤上刪除)。



          3.部署一個web應用

          有兩個辦法可以在系統中部署web服務。

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

          2> 為你的web服務建立一個只包括context內容的XML片斷文件,并把該文件放到$CATALINA_BASE/webapps目錄下。這個web應用本身可以存儲在硬盤上的任何地方。



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



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



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



          <!--

          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片斷提供了一種便利的方法來部署web應用,你不需要編輯server.xml,除非你想改變缺省的部署特性,安裝一個新的web應用時不需要重啟動Tomcat。





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

          關于server.xml中“Host”這個元素,只有在你設置虛擬主機的才需要修改。虛擬主機是一種在一個web服務器上服務多個域名的機制,對每個域名而言,都好象獨享了整個主機。實際上,大多數的小型商務網站都是采用虛擬主機實現的,這主要是因為虛擬主機能直接連接到Internet并提供相應的帶寬,以保障合理的訪問響應速度,另外虛擬主機還能提供一個穩定的固定IP。



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

          在Tomcat中使用虛擬主機,你需要設置DNS或主機數據。為了測試,為本地IP設置一個IP別名就足夠了,接下來,你需要在server.xml中添加幾行內容,如下:



          <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文件,在初始狀態下,只包括一個虛擬主機,但是它容易被擴充到支持多個虛擬主機。在前面的例子中展示的是一個簡單的server.xml版本,其中粗體部分就是用于添加一個虛擬主機。每一個Host元素必須包括一個或多個context元素,所包含的context元素中必須有一個是默認的context,這個默認的context的顯示路徑應該為空(例如,path=””)。



          5.配置基礎驗證(Basic Authentication)

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



          注意:使用Basic Authentication通過被認為是不安全的,因為它沒有強健的加密方法,除非在客戶端和服務器端都使用HTTPS或者其他密碼加密碼方式(比如,在一個虛擬私人網絡中)。若沒有額外的加密方法,網絡管理員將能夠截獲(或濫用)用戶的密碼。但是,如果你是剛開始使用Tomcat,或者你想在你的web應用中測試一下基于容器的安全管理,Basic Authentication還是非常易于設置和使用的。只需要添加<security-constraint>和<login-config>兩個元素到你的web應用的web.xml文件中,并且在CATALINA_BASE/conf/tomcat-users.xml 文件中添加適當的<role>和<user>即可,然后重新啟動Tomcat。



          下面例子中的web.xml摘自一個俱樂部會員網站系統,該系統中只有member目錄被保護起來,并使用Basic Authentication進行身份驗證。請注意,這種方式將有效的代替Apache web服務器中的.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.配置單點登錄(Single Sign-On)

          一旦你設置了realm和驗證的方法,你就需要進行實際的用戶登錄處理。一般說來,對用戶而言登錄系統是一件很麻煩的事情,你必須盡量減少用戶登錄驗證的次數。作為缺省的情況,當用戶第一次請求受保護的資源時,每一個web應用都會要求用戶登錄。如果你運行了多個web應用,并且每個應用都需要進行單獨的用戶驗證,那這看起來就有點像你在與你的用戶搏斗。用戶們不知道怎樣才能把多個分離的應用整合成一個單獨的系統,所有他們也就不知道他們需要訪問多少個不同的應用,只是很迷惑,為什么總要不停的登錄。



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



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

          debug="0"/>



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



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

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

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

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

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

          5> 如果你使用單點登錄,還希望集成一個第三方的web應用到你的網站中來,并且這個新的web應用使用它自己的驗證方式,而不使用容器管理安全,那你基本上就沒招了。你的用戶每次登錄原來所有應用時需要登錄一次,并且在請求新的第三方應用時還得再登錄一次。當然,如果你擁有這個第三方web應用的源碼,而你又是一個程序員,你可以修改它,但那恐怕也不容易做。

          6> 單點登錄需要使用cookies。



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

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



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

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



          Tomcat提供兩種方法在主機上映射這些個人網站,主要使用一對特殊的Listener元素。Listener的className屬性應該是org.apache.catalina.startup.UserConfig,userClass屬性應該是幾個映射類之一。如果你的系統是Unix,它將有一個標準的/etc/passwd文件,該文件中的帳號能夠被運行中的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一樣的目錄下面。當然你也可以改變public_html 到其他任何子目錄下。



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



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

          directoryName="public_html" homeBase="/home"

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



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



          這些Listener元素,如果出現,則必須在Host元素里面,而不能在context元素里面,因為它們都用應用于Host本身。





          8.在Tomcat中使用CGI腳本

          Tomcat主要是作為Servlet/JSP容器,但它也有許多傳統web服務器的性能。支持通用網關接口(Common Gateway Interface,即CGI)就是其中之一,CGI提供一組方法在響應瀏覽器請求時運行一些擴展程序。CGI之所以被稱為通用,是因為它能在大多數程序或腳本中被調用,包括:Perl,Python,awk,Unix shell scripting等,甚至包括Java。當然,你大概不會把一個Java應用程序當作CGI來運行,畢竟這樣太過原始。一般而言,開發Servlet總要比CGI具有更好的效率,因為當用戶點擊一個鏈接或一個按鈕時,你不需要從操作系統層開始進行處理。



          Tomcat包括一個可選的CGI Servlet,允許你運行遺留下來的CGI腳本。



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

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

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

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

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

          5. 重新啟動Tomcat,你的CGI就可以運行了。



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



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

          在Tomcat 4.1(或更高版本,大概),JSP的編譯由包含在Tomcat里面的Ant程序控制器直接執行。這聽起來有一點點奇怪,但這正是Ant有意為之的一部分,有一個API文檔指導開發者在沒有啟動一個新的JVM的情況下,使用Ant。這是使用Ant進行Java開發的一大優勢。另外,這也意味著你現在能夠在Ant中使用任何javac支持的編譯方式,這里有一個關于Apache Ant使用手冊的javac page列表。使用起來是容易的,因為你只需要在<init-param> 元素中定義一個名字叫“compiler”,并且在value中有一個支持編譯的編譯器名字,示例如下:



          <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>



          當然,給出的編譯器必須已經安裝在你的系統中,并且CLASSPATH可能需要設置,那處決于你選擇的是何種編譯器。



          10.限制特定主機訪問(Restricting Access to Specific Hosts)

          有時,你可能想限制對Tomcat web應用的訪問,比如,你希望只有你指定的主機或IP地址可以訪問你的應用。這樣一來,就只有那些指定的的客戶端可以訪問服務的內容了。為了實現這種效果,Tomcat提供了兩個參數供你配置:RemoteHostValve 和RemoteAddrValve。



          通過配置這兩個參數,可以讓你過濾來自請求的主機或IP地址,并允許或拒絕哪些主機/IP。與之類似的,在Apache的httpd文件里有對每個目錄的允許/拒絕指定。

          例如你可以把Admin Web application設置成只允許本地訪問,設置如下:



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

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

          allow="127.0.0.1" deny=""/>

          </Context>



          如果沒有給出允許主機的指定,那么與拒絕主機匹配的主機就會被拒絕,除此之外的都是允許的。與之類似,如果沒有給出拒絕主機的指定,那么與允許主機匹配的主機就會被允許,除此之外的都是拒絕的。



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



          作者簡介:

          Jason Brittain是CollabNet公司的一名資深軟件工程師,主要負責軟件底層架構的開發。他已經為Apache Jakarta項目做了很多貢獻,多年以來,他一直是一名積極的開源軟件開發者。



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



          譯者簡介:

          陳光(Holen Chen),J2EE項目經理,熟悉知識管理及電子政務,致力于Apache Jakarta項目在國廣的推廣及深層次應用,可通過 holen@263.net與作者聯系。

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

          下面給出得Log4J配置文件實現了輸出到控制臺,文件,回滾文件,發送日志郵件,輸出到數據庫日志表,自定義標簽等全套功能。

          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 閱讀(250) | 評論 (0)編輯 收藏

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

          ?

          #?應用于控制臺

          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


          #應用于文件

          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



          #?應用于文件回滾

          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
          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



          #?發送日志給郵件

          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



          #?用于數據庫
          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 閱讀(354) | 評論 (0)編輯 收藏

          僅列出標題
          共19頁: First 上一頁 7 8 9 10 11 12 13 14 15 下一頁 Last 
          主站蜘蛛池模板: 马鞍山市| 中阳县| 来凤县| 正安县| 汝南县| 行唐县| 台山市| 怀集县| 石景山区| 天气| 牡丹江市| 广西| 阿勒泰市| 霍城县| 海晏县| 大石桥市| 毕节市| 锦州市| 民乐县| 惠安县| 拉萨市| 杭州市| 老河口市| 和静县| 顺义区| 苏尼特左旗| 天峻县| 务川| 中卫市| 玉树县| 景泰县| 紫阳县| 宜川县| 云浮市| 南川市| 浦江县| 中阳县| 嘉义市| 长沙市| 铁岭县| 昭觉县|