JAVA的newInstance()和new的區別(JAVA反射機制,通過類名來獲取該類的實例化對象)

1.在初始化一個類,生成一個實例的時候;newInstance() 和 new 有什麼區別?
     用newInstance與用new是區別的,區別在於創建對象的方式不一樣,前者是使用類加載機制,那麼爲什麼會有兩種創建對象方式?這個就要從可伸縮、可擴展,可重用等軟件思想上解釋了。
     Java中工廠模式經常使用newInstance來創建對象,因此從爲什麼要使用工廠模式上也可以找到具體答案。

案例:
     Class c = Class.forName(“A”);factory = (AInterface)c.newInstance();
     其中AInterface是A的接口,如果下面這樣寫,你可能會理解:
     String className = “A”;Class c = Class.forName(className);factory = (AInterface)c.newInstance();
     進一步,如果下面寫,你可能會理解:
     String className = readfromXMlConfig;//從xml 配置文件中獲得字符串Class c = Class.forName(className);factory = (AInterface)c.newInstance();
     上面代碼就消滅了A類名稱,優點:無論A類怎麼變化,上述代碼不變,甚至可以更換A的兄弟類B , C , D….等,只要他們繼承Ainterface就可以。
     從jvm的角度看,我們使用new的時候,這個要new的類可以沒有加載;
     但是使用newInstance時候,就必須保證:1、這個類已經加載;2、這個類已經連接了。而完成上面兩個步驟的正是class的靜態方法forName()方法,這個靜態方法調用了啓動類加載器(就是加載javaAPI的那個加載器)。
有了上面jvm上的理解,那麼我們可以這樣說,newInstance實際上是把new這個方式分解爲兩步,即,首先調用class的加載方法加載某個類,然後實例化。
     這樣分步的好處是顯而易見的。我們可以在調用class的靜態加載方法forName時獲得更好的靈活性,提供給了我們降耦的手段。

[補充:]
newInstance: 弱類型。低效率。只能調用無參構造。
new: 強類型。相對高效。能調用任何public構造。
newInstance()是實現IOC、反射、面對接口編程 和 依賴倒置 等技術方法的必然選擇,new 只能實現具體類的實例化,不適合於接口編程。
裏面就是通過這個類的默認構造函數構建了一個對象,如果沒有默認構造函數就拋出InstantiationException, 如果沒有訪問默認構造函數的權限就拋出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();
        }
    }
}

//通過反射獲取該類的實例化對象,並由接口來調用該類的方法
這樣實現的好處就是如果也有一個classB類實現了這個接口的方法。
public class classB implements duang{
	@Override
	public String getClazzName() {
		return "B類"+this.getClass().getName();
	}
}

直接把
Class clazz = Class.forName("classA");
換成
Class clazz = Class.forName("classB");



反射的各種應用:參考這個鏈接
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章