理解Class.forname

Java開發特別是數據庫開發中,經常會用到Class.forName( )這個方法。通過查詢Java Documentation我們會發現使用Class.forName( )靜態方法的目的是爲了動態加載類。在加載完成後,一般還要調用Class下的newInstance( )靜態方法來實例化對象以便操作。

爲什麼要使用class.forName("").newInstance()class.forName().newInstance()它和new之間有什麼區別呢?

1 我們都知道,java中任何class都要裝載在虛擬機上才能運行,而Class.forName" ")就是當給定一個代表包名和類名的字符串時,用來實例化這個類的。如:

A a = (A)Class.forName("pacage.A").newInstance();

和我們常用的:

A a=new A()

的效果是一樣的,都是返回一個指向類A的對象(object

Class.forName("pacage.A")則返回的是一個類(classA

但爲什麼在jdbc連接數據庫的寫法裏是有的是Class.forNamexxx.xx.xx)另一些是:Class.forNamexxx.xx.xx. newInstance()

首要說明的是:使用第一種是最確切的。

Class.forNamexxx.xx.xx)的作用是要求JVM查找並加載指定的類,也就是說JVM會執行該類的靜態代碼段。

JDBC規範中明確要求這個Driver類必須向DriverManager註冊自己,即任何一個JDBC DriverDriver類的代碼都必須類似如下:
  

  所以我們在使用JDBC時只需要Class.forName(XXX.XXX);就可以了

JDBC驅動中,有一塊靜態代碼,也叫靜態初始化塊,它執行的時間是當class調入到內存中就執行(你可以想像成,當類調用到內存後就執行一個方法)。所以很多人把jdbc driver調入到內存中,再實例化對象是沒有意義的

 

2

      newInstance()方法和new關鍵字除了一個是方法,一個是關鍵字外,它們的主要在於創建對象的方式不一樣,前者是使用類加載機制,後者是創建一個新類。

那麼爲什麼會有兩種創建對象方式?這主要考慮到軟件的可伸縮、可擴展和可重用等軟件設計思想。

Java中工廠模式經常使用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構造。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章