newInstance() 和 new 有什么區(qū)別?(轉(zhuǎn))
在初始化一個(gè)類(lèi),生成一個(gè)實(shí)例的時(shí)候;newInstance() 和 new 有什么區(qū)別?用newInstance與用new是區(qū)別的,區(qū)別在于創(chuàng)建對(duì)象的方式不一樣,前者是使用類(lèi)加載機(jī)制,那么為什么會(huì)有兩種創(chuàng)建對(duì)象方式?這個(gè)就要從可伸縮、可擴(kuò)展,可重用等軟件思想上解釋了。
Java中工廠模式經(jīng)常使用newInstance來(lái)創(chuàng)建對(duì)象,因此從為什么要使用工廠模式上也可以找到具體答案。
例如:
Class c = Class.forName(“A”);factory = (AInterface)c.newInstance();
其中AInterface是A的接口,如果下面這樣寫(xiě),你可能會(huì)理解:
String className = "A";Class c = Class.forName(className);factory = (AInterface)c.newInstance();
進(jìn)一步,如果下面寫(xiě),你可能會(huì)理解:
String className = readfromXMlConfig;//從xml 配置文件中獲得字符串Class c = Class.forName(className);factory = (AInterface)c.newInstance();
上面代碼就消滅了A類(lèi)名稱,優(yōu)點(diǎn):無(wú)論A類(lèi)怎么變化,上述代碼不變,甚至可以更換A的兄弟類(lèi)B , C , D....等,只要他們繼承Ainterface就可以。
從jvm的角度看,我們使用new的時(shí)候,這個(gè)要new的類(lèi)可以沒(méi)有加載;
但是使用newInstance時(shí)候,就必須保證:1、這個(gè)類(lèi)已經(jīng)加載;2、這個(gè)類(lèi)已經(jīng)連接了。而完成上面兩個(gè)步驟的正是class的靜態(tài)方法forName()方法,這個(gè)靜態(tài)方法調(diào)用了啟動(dòng)類(lèi)加載器(就是加載java API的那個(gè)加載器)。
有了上面jvm上的理解,那么我們可以這樣說(shuō),newInstance實(shí)際上是把new這個(gè)方式分解為兩步,即,首先調(diào)用class的加載方法加載某個(gè)類(lèi),然后實(shí)例化。
這樣分步的好處是顯而易見(jiàn)的。我們可以在調(diào)用class的靜態(tài)加載方法forName時(shí)獲得更好的靈活性,提供給了我們降耦的手段。
[補(bǔ)充:]
newInstance: 弱類(lèi)型。低效率。只能調(diào)用無(wú)參構(gòu)造。
new: 強(qiáng)類(lèi)型。相對(duì)高效。能調(diào)用任何public構(gòu)造。
newInstance()是實(shí)現(xiàn)IOC、反射、面對(duì)接口編程 和 依賴倒置 等技術(shù)方法的必然選擇,new 只能實(shí)現(xiàn)具體類(lèi)的實(shí)例化,不適合于接口編程。
里面就是通過(guò)這個(gè)類(lèi)的默認(rèn)構(gòu)造函數(shù)構(gòu)建了一個(gè)對(duì)象,如果沒(méi)有默認(rèn)構(gòu)造函數(shù)就拋出InstantiationException, 如果沒(méi)有訪問(wèn)默認(rèn)構(gòu)造函數(shù)的權(quán)限就拋出IllegalAccessException
posted on 2011-04-14 17:37 lau 閱讀(1073) 評(píng)論(0) 編輯 收藏 所屬分類(lèi): J2SE