在Java開(kāi)發(fā)特別是數(shù)據(jù)庫(kù)開(kāi)發(fā)中爸吮,經(jīng)常會(huì)用到Class.forName( )這個(gè)方法某饰。通過(guò)查詢Java Documentation我們會(huì)發(fā)現(xiàn)使用Class.forName( )靜態(tài)方法的目的是為了動(dòng)態(tài)加載類抬旺。在加載完成后渗磅,一般還要調(diào)用Class下的newInstance( )靜態(tài)方法來(lái)實(shí)例化對(duì)象以便操作颁湖。因此,單單使用Class.forName( )是動(dòng)態(tài)加載類是沒(méi)有用的搁胆,其最終目的是為了實(shí)例化對(duì)象弥搞。
這里有必要提一下就是Class下的newInstance()和new有什么區(qū)別?渠旁,首先攀例,newInstance( )是一個(gè)方法,而new是一個(gè)關(guān)鍵字顾腊,其次粤铭,Class下的newInstance()的使用有局限,因?yàn)樗蓪?duì)象只能調(diào)用無(wú)參的構(gòu)造函數(shù)杂靶,而使用new關(guān)鍵字生成對(duì)象沒(méi)有這個(gè)限制梆惯。
好,到此為止吗垮,我們總結(jié)如下:
- Class.forName("")返回的是類
- Class.forName("").newInstance()返回的是object
有數(shù)據(jù)庫(kù)開(kāi)發(fā)經(jīng)驗(yàn)朋友會(huì)發(fā)現(xiàn)垛吗,為什么在我們加載數(shù)據(jù)庫(kù)驅(qū)動(dòng)包的時(shí)候有的卻沒(méi)有調(diào)用newInstance( )方法呢?即有的jdbc連接數(shù)據(jù)庫(kù)的寫法里是
Class.forName(xxx.xx.xx);而有一些:Class.forName(xxx.xx.xx).newInstance()烁登,為什么會(huì)有這兩種寫法呢怯屉?
剛才提到,Class.forName("");的作用是要求JVM查找并加載指定的類饵沧,如果在類中有靜態(tài)初始化器的話锨络,JVM必然會(huì)執(zhí)行該類的靜態(tài)代碼段。而在JDBC規(guī)范中明確要求這個(gè)Driver類必須向DriverManager注冊(cè)自己狼牺,即任何一個(gè)JDBC Driver的Driver類的代碼都必須類似如下:
public class MyJDBCDriver implements Driver {
static {
DriverManager.registerDriver(new MyJDBCDriver());
}
}
既然在靜態(tài)初始化器的中已經(jīng)進(jìn)行了注冊(cè)羡儿,所以我們?cè)谑褂肑DBC時(shí)只需要Class.forName(XXX.XXX);就可以了。
Java中工廠模式經(jīng)常使用newInstance()方法來(lái)創(chuàng)建對(duì)象是钥,因此從為什么要使用工廠模式上可以找到具體答案掠归。 例如:
class c = Class.forName("Example");
factory = (ExampleInterface)c.newInstance();
其中ExampleInterface是Example的接口,可以寫成如下形式:
String className = "Example";
class c = Class.forName(className);
factory = (ExampleInterface)c.newInstance();
進(jìn)一步可以寫成如下形式:
String className = readfromXMlConfig;//從xml 配置文件中獲得字符串
class c = Class.forName(className);
factory = (ExampleInterface)c.newInstance();
上面代碼已經(jīng)不存在Example的類名稱咏瑟,它的優(yōu)點(diǎn)是拂到,無(wú)論Example類怎么變化,上述代碼不變码泞,甚至可以更換Example的兄弟類Example2 , Example3 , Example4……兄旬,只要他們繼承ExampleInterface就可以。
從JVM的角度看余寥,我們使用關(guān)鍵字new創(chuàng)建一個(gè)類的時(shí)候领铐,這個(gè)類可以沒(méi)有被加載。但是使用newInstance()方法的時(shí)候宋舷,就必須保證:1绪撵、這個(gè) 類已經(jīng)加載;2祝蝠、這個(gè)類已經(jīng)連接了音诈。而完成上面兩個(gè)步驟的正是Class的靜態(tài)方法forName()所完成的幻碱,這個(gè)靜態(tài)方法調(diào)用了啟動(dòng)類加載器,即加載 java API的那個(gè)加載器细溅。
現(xiàn)在可以看出褥傍,newInstance()實(shí)際上是把new這個(gè)方式分解為兩步,即首先調(diào)用Class加載方法加載某個(gè)類喇聊,然后實(shí)例化恍风。 這樣分步的好處是顯而易見(jiàn)的。我們可以在調(diào)用class的靜態(tài)加載方法forName時(shí)獲得更好的靈活性誓篱,提供給了一種降耦的手段朋贬。
最后用最簡(jiǎn)單的描述來(lái)區(qū)分new關(guān)鍵字和newInstance()方法的區(qū)別:
- newInstance: 弱類型。低效率窜骄。只能調(diào)用無(wú)參構(gòu)造锦募。
- new: 強(qiáng)類型。相對(duì)高效邻遏。能調(diào)用任何public構(gòu)造御滩。