一個名字可以被用來引用位于不同包內(nèi)的多個類。下面的程序就是在探究當(dāng)你重用了一個平臺類的名字時(shí),會發(fā)生什么。你認(rèn)為它會做些什么呢?盡管這個程序?qū)儆谀欠N讓你通常一看到就會感到尷尬的程序,但是你還是應(yīng)該繼續(xù)下去,把門鎖上,把百葉窗拉上,然后試試看:
public class StrungOut {
public static void main(String[] args) {
String s = new String("Hello world");
System.out.println(s);
}
}
class String {
private final java.lang.String s;
public String(java.lang.String s) {
this.s = s;
}
public java.lang.String toString() {
return s;
}
}
如果說這個程序有點(diǎn)讓人討厭的話,它看起來還是相當(dāng)簡單的。在未命名包中的String類就是一個java.lang.String實(shí)例的包裝器,看起來該程序應(yīng)該打印Hello world。如果你嘗試著運(yùn)行該程序,你會發(fā)現(xiàn)你運(yùn)行不了它,VM將彈出了一個像下面這樣的錯誤消息:
Exception in thread "main" java.lang.NoSuchMethodError: main
但是它肯定是一個main方法的:它就白紙黑字地寫在那里。為什么VM找不到它呢?
VM不能找到main方法是因?yàn)樗⒉辉谀抢铩1M管StrungOut有一個被命名為main的方法,但是它卻具有錯誤的簽名。一個main方法必須接受一個單一的字符串?dāng)?shù)組參數(shù)[JVMS 5.2]。VM努力要告訴我們的是StrungOut.main接受的是由我們的String類所構(gòu)成的數(shù)組,它無論如何都與java.lang.String沒有任何關(guān)系。
如果你確實(shí)需要編寫自己的字符串類,看在老天爺?shù)姆萆?,千萬不要稱其為String。要避免重用平臺類的名字,并且千萬不要重用java.lang中的類名,因?yàn)檫@些名字會被各處的程序自動加載。程序員習(xí)慣于看到這些名字以無限定的形式出現(xiàn),并且會很自然地認(rèn)為這些名字引用的是我們所熟知的java.lang中的類。如果你重用了這些名字的某一個,那么當(dāng)這個名字在其自己的包內(nèi)被使用時(shí),該名字的無限定形式將會引用到新的定義上。
要訂正該程序,只需為這個非標(biāo)準(zhǔn)的字符串類挑選一個合理的名字即可。該程序下面的這個版本很明顯是正確的,而且它比最初的版本要更易于理解。它將打印出如你所期望的Hello World:
public class StrungOut {
public static void main(String[ ] args) {
MyString s = new MyString("Hello world");
System.out.println(s);
}
}
class MyString {
private final java.lang.String s;
public MyString(java.lang.String s) { this.s = s;}
public java.lang.String toString() { return s;}
}
寬泛地講,本謎題的教訓(xùn)就是要避免重用類名,尤其是Java平臺類的類名。千萬不要重用java.lang包內(nèi)的類名,相同的教訓(xùn)也適用于類庫的設(shè)計(jì)者。Java平臺的設(shè)計(jì)者已經(jīng)在這個問題上栽過數(shù)次了,的例子有java.sql.Date,它與java.util.Date和org.omg.CORBA.Object相沖突。與在本章中的許多其他謎題一樣,這個教訓(xùn)是有關(guān)你在除了覆寫之外的其他情況應(yīng)該避免名字重用這一原則的一個具體實(shí)例。對平臺實(shí)現(xiàn)者來說,其教訓(xùn)是診斷信息應(yīng)該清晰地解釋失敗的原因。VM應(yīng)該可以很容易地將沒有任何具有正確簽名的main方法的情況與根本就沒有任何main方法的情況區(qū)分開。
public class StrungOut {
public static void main(String[] args) {
String s = new String("Hello world");
System.out.println(s);
}
}
class String {
private final java.lang.String s;
public String(java.lang.String s) {
this.s = s;
}
public java.lang.String toString() {
return s;
}
}
如果說這個程序有點(diǎn)讓人討厭的話,它看起來還是相當(dāng)簡單的。在未命名包中的String類就是一個java.lang.String實(shí)例的包裝器,看起來該程序應(yīng)該打印Hello world。如果你嘗試著運(yùn)行該程序,你會發(fā)現(xiàn)你運(yùn)行不了它,VM將彈出了一個像下面這樣的錯誤消息:
Exception in thread "main" java.lang.NoSuchMethodError: main
但是它肯定是一個main方法的:它就白紙黑字地寫在那里。為什么VM找不到它呢?
VM不能找到main方法是因?yàn)樗⒉辉谀抢铩1M管StrungOut有一個被命名為main的方法,但是它卻具有錯誤的簽名。一個main方法必須接受一個單一的字符串?dāng)?shù)組參數(shù)[JVMS 5.2]。VM努力要告訴我們的是StrungOut.main接受的是由我們的String類所構(gòu)成的數(shù)組,它無論如何都與java.lang.String沒有任何關(guān)系。
如果你確實(shí)需要編寫自己的字符串類,看在老天爺?shù)姆萆?,千萬不要稱其為String。要避免重用平臺類的名字,并且千萬不要重用java.lang中的類名,因?yàn)檫@些名字會被各處的程序自動加載。程序員習(xí)慣于看到這些名字以無限定的形式出現(xiàn),并且會很自然地認(rèn)為這些名字引用的是我們所熟知的java.lang中的類。如果你重用了這些名字的某一個,那么當(dāng)這個名字在其自己的包內(nèi)被使用時(shí),該名字的無限定形式將會引用到新的定義上。
要訂正該程序,只需為這個非標(biāo)準(zhǔn)的字符串類挑選一個合理的名字即可。該程序下面的這個版本很明顯是正確的,而且它比最初的版本要更易于理解。它將打印出如你所期望的Hello World:
public class StrungOut {
public static void main(String[ ] args) {
MyString s = new MyString("Hello world");
System.out.println(s);
}
}
class MyString {
private final java.lang.String s;
public MyString(java.lang.String s) { this.s = s;}
public java.lang.String toString() { return s;}
}
寬泛地講,本謎題的教訓(xùn)就是要避免重用類名,尤其是Java平臺類的類名。千萬不要重用java.lang包內(nèi)的類名,相同的教訓(xùn)也適用于類庫的設(shè)計(jì)者。Java平臺的設(shè)計(jì)者已經(jīng)在這個問題上栽過數(shù)次了,的例子有java.sql.Date,它與java.util.Date和org.omg.CORBA.Object相沖突。與在本章中的許多其他謎題一樣,這個教訓(xùn)是有關(guān)你在除了覆寫之外的其他情況應(yīng)該避免名字重用這一原則的一個具體實(shí)例。對平臺實(shí)現(xiàn)者來說,其教訓(xùn)是診斷信息應(yīng)該清晰地解釋失敗的原因。VM應(yīng)該可以很容易地將沒有任何具有正確簽名的main方法的情況與根本就沒有任何main方法的情況區(qū)分開。