下面的程序被設計用來打印它的類文件的名稱。如果你不熟悉類字面常量,那么我告訴你Me.class.getName()將返回Me類完整的名稱,即“com.javapuzzlers.Me”。那么,這個程序會打印出什么呢?
package com.javapuzzlers;
public class Me {
public static void main(String[] args){
System.out.println(
Me.class.getName().
replaceAll(".","/") + ".class");
}
}
該程序看起來會獲得它的類名(“com.javapuzzlers.Me”),然后用“/”替換掉所有出現的字符串“.”,并在末尾追加字符串“.class”。你可能會認為該程序將打印com/javapuzzlers/Me.class,該程序正式從這個類文件中被加載的。如果你運行這個程序,就會發現它實際上打印的是///////////////////.class。到底怎么回事?難道我們是斜杠的受害者嗎?
問題在于String.replaceAll接受了一個正則表達式作為它的第一個參數,而并非接受了一個字符序列字面常量。(正則表達式已經被添加到了Java平臺的1.4版本中。)正則表達式“.”可以匹配任何單個的字符,因此,類名中的每一個字符都被替換成了一個斜杠,進而產生了我們看到的輸出。
要想只匹配句點符號,在正則表達式中的句點必須在其前面添加一個反斜杠(")進行轉義。因為反斜杠字符在字面含義的字符串中具有特殊的含義——它標識轉義字符序列的開始——因此反斜杠自身必須用另一個反斜杠來轉義,這樣就可以產生一個轉義字符序列,它可以在字面含義的字符串中生成一個反斜杠。把這些合在一起,就可以使下面的程序打印出我們所期望的com/javapuzzlers/Me.class:
package com.javapuzzlers;
public class Me {
public static void main(String[] args){
System.out.println(
Me.class.getName().replaceAll(""".","/") + ".class");
}
}
為了解決這類問題,5.0版本提供了新的靜態方法java.util.regex.Pattern.quote。它接受一個字符串作為參數,并可以添加必需的轉義字符,它將返回一個正則表達式字符串,該字符串將精確匹配輸入的字符串。下面是使用該方法之后的程序:
package com.javapuzzlers;
import java.util.regex.Pattern;
public class Me {
public static void main(String[] args){
System.out.println(Me.class.getName().
replaceAll(Pattern.quote("."),"/") + ".class");
}
}
該程序的另一個問題是:其正確的行為是與平臺相關的。并不是所有的文件系統都使用斜杠符號來分隔層次結構的文件名組成部分的。要想獲取一個你正在運行的平臺上的有效文件名,你應該使用正確的平臺相關的分隔符號來代替斜杠符號。這正是下一個謎題所要做的。