java 反射 new区别_JAVA的newInstance()和new的区别(JAVA反射机制,通过类名来获取该类的实例化对象)...
1.在初始化一個(gè)類,生成一個(gè)實(shí)例的時(shí)候;newInstance() 和 new 有什么區(qū)別?
用newInstance與用new是區(qū)別的,區(qū)別在于創(chuàng)建對(duì)象的方式不一樣,前者是使用類加載機(jī)制,那么為什么會(huì)有兩種創(chuàng)建對(duì)象方式?這個(gè)就要從可伸縮、可擴(kuò)展,可重用等軟件思想上解釋了。
Java中工廠模式經(jīng)常使用newInstance來創(chuàng)建對(duì)象,因此從為什么要使用工廠模式上也可以找到具體答案。
案例:
Class c = Class.forName(“A”);factory = (AInterface)c.newInstance();
其中AInterface是A的接口,如果下面這樣寫,你可能會(huì)理解:
String className = “A”;Class c = Class.forName(className);factory = (AInterface)c.newInstance();
進(jìn)一步,如果下面寫,你可能會(huì)理解:
String className = readfromXMlConfig;//從xml 配置文件中獲得字符串Class c = Class.forName(className);factory = (AInterface)c.newInstance();
上面代碼就消滅了A類名稱,優(yōu)點(diǎn):無論A類怎么變化,上述代碼不變,甚至可以更換A的兄弟類B , C , D….等,只要他們繼承Ainterface就可以。
從jvm的角度看,我們使用new的時(shí)候,這個(gè)要new的類可以沒有加載;
但是使用newInstance時(shí)候,就必須保證:1、這個(gè)類已經(jīng)加載;2、這個(gè)類已經(jīng)連接了。而完成上面兩個(gè)步驟的正是class的靜態(tài)方法forName()方法,這個(gè)靜態(tài)方法調(diào)用了啟動(dòng)類加載器(就是加載javaAPI的那個(gè)加載器)。
有了上面jvm上的理解,那么我們可以這樣說,newInstance實(shí)際上是把new這個(gè)方式分解為兩步,即,首先調(diào)用class的加載方法加載某個(gè)類,然后實(shí)例化。
這樣分步的好處是顯而易見的。我們可以在調(diào)用class的靜態(tài)加載方法forName時(shí)獲得更好的靈活性,提供給了我們降耦的手段。
[補(bǔ)充:]
newInstance: 弱類型。低效率。只能調(diào)用無參構(gòu)造。
new: 強(qiáng)類型。相對(duì)高效。能調(diào)用任何public構(gòu)造。
newInstance()是實(shí)現(xiàn)IOC、反射、面對(duì)接口編程 和 依賴倒置 等技術(shù)方法的必然選擇,new 只能實(shí)現(xiàn)具體類的實(shí)例化,不適合于接口編程。
里面就是通過這個(gè)類的默認(rèn)構(gòu)造函數(shù)構(gòu)建了一個(gè)對(duì)象,如果沒有默認(rèn)構(gòu)造函數(shù)就拋出InstantiationException, 如果沒有訪問默認(rèn)構(gòu)造函數(shù)的權(quán)限就拋出IllegalAccessException
public?interface?duang?{
public?String?getClazzName();
}
public?class?classA?implements?duang{
@Override
public?String?getClazzName()?{
return?"A類"+this.getClass().getName();
}
}
public?class?classD?{
public?static?void?main(String[]?args)
{
//????????System.out.println(classA.getClazzName());
try?{
Class?clazz?=?Class.forName("classA");
duang?duang?=?(duang)?clazz.newInstance();
System.out.println(duang.getClazzName());
}?catch?(ClassNotFoundException?e)?{
e.printStackTrace();
}?catch?(InstantiationException?e)?{
e.printStackTrace();
}?catch?(IllegalAccessException?e)?{
e.printStackTrace();
}
}
}
//通過反射獲取該類的實(shí)例化對(duì)象,并由接口來調(diào)用該類的方法
這樣實(shí)現(xiàn)的好處就是如果也有一個(gè)classB類實(shí)現(xiàn)了這個(gè)接口的方法。
public?class?classB?implements?duang{
@Override
public?String?getClazzName()?{
return?"B類"+this.getClass().getName();
}
}
直接把
Class?clazz?=?Class.forName("classA");
換成
Class?clazz?=?Class.forName("classB");
反射的各種應(yīng)用:參考這個(gè)鏈接
與50位技術(shù)專家面對(duì)面20年技術(shù)見證,附贈(zèng)技術(shù)全景圖總結(jié)
以上是生活随笔為你收集整理的java 反射 new区别_JAVA的newInstance()和new的区别(JAVA反射机制,通过类名来获取该类的实例化对象)...的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: mac java偏好设置_在 Mac中配
- 下一篇: java看log技巧_Log日志框架的学