下面的程序被設(shè)計用來打印它的類文件的名稱。如果你不熟悉類字面常量,那么我告訴你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”),然后用“/”替換掉所有出現(xiàn)的字符串“.”,并在末尾追加字符串“.class”。你可能會認(rèn)為該程序?qū)⒋蛴om/javapuzzlers/Me.class,該程序正式從這個類文件中被加載的。如果你運(yùn)行這個程序,就會發(fā)現(xiàn)它實(shí)際上打印的是///////////////////.class。到底怎么回事?難道我們是斜杠的受害者嗎?
問題在于String.replaceAll接受了一個正則表達(dá)式作為它的第一個參數(shù),而并非接受了一個字符序列字面常量。(正則表達(dá)式已經(jīng)被添加到了Java平臺的1.4版本中。)正則表達(dá)式“.”可以匹配任何單個的字符,因此,類名中的每一個字符都被替換成了一個斜杠,進(jìn)而產(chǎn)生了我們看到的輸出。
要想只匹配句點(diǎn)符號,在正則表達(dá)式中的句點(diǎn)必須在其前面添加一個反斜杠(\)進(jìn)行轉(zhuǎn)義。因?yàn)榉葱备茏址谧置婧x的字符串中具有特殊的含義——它標(biāo)識轉(zhuǎn)義字符序列的開始——因此反斜杠自身必須用另一個反斜杠來轉(zhuǎn)義,這樣就可以產(chǎn)生一個轉(zhuǎn)義字符序列,它可以在字面含義的字符串中生成一個反斜杠。把這些合在一起,就可以使下面的程序打印出我們所期望的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版本提供了新的靜態(tài)方法java.util.regex.Pattern.quote。它接受一個字符串作為參數(shù),并可以添加必需的轉(zhuǎn)義字符,它將返回一個正則表達(dá)式字符串,該字符串將精確匹配輸入的字符串。下面是使用該方法之后的程序:
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");
}
}
該程序的另一個問題是:其正確的行為是與平臺相關(guān)的。并不是所有的文件系統(tǒng)都使用斜杠符號來分隔層次結(jié)構(gòu)的文件名組成部分的。要想獲取一個你正在運(yùn)行的平臺上的有效文件名,你應(yīng)該使用正確的平臺相關(guān)的分隔符號來代替斜杠符號。這正是下一個謎題所要做的。
package com.javapuzzlers;
public class Me {
public static void main(String[] args){
System.out.println(
Me.class.getName().
replaceAll(".","/") + ".class");
}
}
該程序看起來會獲得它的類名(“com.javapuzzlers.Me”),然后用“/”替換掉所有出現(xiàn)的字符串“.”,并在末尾追加字符串“.class”。你可能會認(rèn)為該程序?qū)⒋蛴om/javapuzzlers/Me.class,該程序正式從這個類文件中被加載的。如果你運(yùn)行這個程序,就會發(fā)現(xiàn)它實(shí)際上打印的是///////////////////.class。到底怎么回事?難道我們是斜杠的受害者嗎?
問題在于String.replaceAll接受了一個正則表達(dá)式作為它的第一個參數(shù),而并非接受了一個字符序列字面常量。(正則表達(dá)式已經(jīng)被添加到了Java平臺的1.4版本中。)正則表達(dá)式“.”可以匹配任何單個的字符,因此,類名中的每一個字符都被替換成了一個斜杠,進(jìn)而產(chǎn)生了我們看到的輸出。
要想只匹配句點(diǎn)符號,在正則表達(dá)式中的句點(diǎn)必須在其前面添加一個反斜杠(\)進(jìn)行轉(zhuǎn)義。因?yàn)榉葱备茏址谧置婧x的字符串中具有特殊的含義——它標(biāo)識轉(zhuǎn)義字符序列的開始——因此反斜杠自身必須用另一個反斜杠來轉(zhuǎn)義,這樣就可以產(chǎn)生一個轉(zhuǎn)義字符序列,它可以在字面含義的字符串中生成一個反斜杠。把這些合在一起,就可以使下面的程序打印出我們所期望的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版本提供了新的靜態(tài)方法java.util.regex.Pattern.quote。它接受一個字符串作為參數(shù),并可以添加必需的轉(zhuǎn)義字符,它將返回一個正則表達(dá)式字符串,該字符串將精確匹配輸入的字符串。下面是使用該方法之后的程序:
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");
}
}
該程序的另一個問題是:其正確的行為是與平臺相關(guān)的。并不是所有的文件系統(tǒng)都使用斜杠符號來分隔層次結(jié)構(gòu)的文件名組成部分的。要想獲取一個你正在運(yùn)行的平臺上的有效文件名,你應(yīng)該使用正確的平臺相關(guān)的分隔符號來代替斜杠符號。這正是下一個謎題所要做的。