天天看點

反射機制知識點(new和newInstance之差別)

在Java開發特别是資料庫開發中,經常會用到Class.forName( )這個方法。通過查詢Java Documentation我們會發現使用Class.forName( )靜态方法的目的是為了動态加載類。在加載完成後,一般還要調用Class下的newInstance( )靜态方法來執行個體化對象以便操作。是以,單單使用Class.forName( )是動态加載類是沒有用的,其最終目的是為了執行個體化對象。

這裡有必要提一下就是Class下的newInstance()和new有什麼差別?,首先,newInstance( )是一個方法,而new是一個關鍵字,其次,Class下的newInstance()的使用有局限,因為它生成對象隻能調用無參的構造函數,而使用new關鍵字生成對象沒有這個限制。

好,到此為止,我們總結如下:

Class.forName(“”)傳回的是類

Class.forName(“”).newInstance()傳回的是object

有資料庫開發經驗朋友會發現,為什麼在我們加載資料庫驅動包的時候有的卻沒有調用newInstance( )方法呢?即有的jdbc連接配接資料庫的寫法裡是Class.forName(xxx.xx.xx);有的是Class.forName(xxx.xx.xx).newInstance(),為什麼會有這兩種寫法呢?

剛才提到,Class.forName(“”);的作用是要求JVM查找并加載指定的類,如果在類中有靜态初始化器的話,JVM必然會執行該類的靜态代碼段。而在JDBC規範中明确要求這個Driver類必須向DriverManager注冊自己,即任何一個JDBC Driver的Driver類的代碼都必須類似如下:

public class MyJDBCDriver implements Driver {
static 
    {
        DriverManager.registerDriver(new MyJDBCDriver());
    }
}
           

既然在靜态初始化器的中已經進行了注冊,是以我們在使用JDBC時隻需要Class.forName(XXX.XXX);就可以了。

Java中工廠模式經常使用newInstance()方法來建立對象,是以從為什麼要使用工廠模式上可以找到具體答案。 例如:

class c = Class.forName(“Example”);
factory = (ExampleInterface)c.newInstance();
           

其中ExampleInterface是Example的接口,可以寫成如下形式:

String className = "Example";
class c = Class.forName(className);
factory = (ExampleInterface)c.newInstance();
           

進一步可以寫成如下形式:

String className = readfromXMlConfig;//從xml 配置檔案中獲得字元串
class c = Class.forName(className);
factory = (ExampleInterface)c.newInstance();
           

上面代碼已經不存在Example的類名稱,它的優點是,無論Example類怎麼變化,上述代碼不變,甚至可以更換Example的兄弟類Example2 , Example3 , Example4……,隻要他們繼承ExampleInterface就可以。

從JVM的角度看,我們使用關鍵字new建立一個類的時候,這個類可以沒有被加載。但是使用newInstance()方法的時候,就必須保證:1、這個 類已經加載;2、這個類已經連接配接了。而完成上面兩個步驟的正是Class的靜态方法forName()所完成的,這個靜态方法調用了啟動類加載器,即加載 java API的那個加載器。

現在可以看出,newInstance()實際上是把new這個方式分解為兩步,即首先調用Class加載方法加載某個類,然後執行個體化。 這樣分步的好處是顯而易見的。我們可以在調用class的靜态加載方法forName時獲得更好的靈活性,提供給了一種降耦的手段。

最後用最簡單的描述來區分new關鍵字和newInstance()方法的差別:

newInstance: 弱類型。低效率。隻能調用無參構造

new: 強類型。相對高效。能調用任何public構造

<script type="text/javascript"> $(function () { $('pre.prettyprint code').each(function () { var lines = $(this).text().split('\n').length; var $numbering = $('<ul/>').addClass('pre-numbering').hide(); $(this).addClass('has-numbering').parent().append($numbering); for (i = 1; i <= lines; i++) { $numbering.append($('<li/>').text(i)); }; $numbering.fadeIn(1700); }); }); </script>