天天看點

Class.forName()用法詳解

主要功能
Class.forName(xxx.xx.xx)傳回的是一個類。
Class.forName(xxx.xx.xx)的作用是要求JVM查找并加載指定的類,也就是說JVM會執行該類的靜态代碼段。      

下面,通過解答以下三個問題的來詳細講解下Class.forName()的用法。

一.什麼時候用Class.forName()?

先來個熱身,給你一個字元串變量,它代表一個類的包名和類名,你怎麼執行個體化它?你第一想到的肯定是new,但是注意一點:

A a = (A)Class.forName(“pacage.A”).newInstance();

這和你 A a = new A(); 是一樣的效果。

現在言歸正傳。

動态加載和建立Class 對象,比如想根據使用者輸入的字元串來建立對象時需要用到:

String str = “使用者輸入的字元串” ;

Class t = Class.forName(str);

t.newInstance();

在初始化一個類,生成一個執行個體的時候,newInstance()方法和new關鍵字除了一個是方法,一個是關鍵字外,最主要有什麼差別?

它們的差別在于建立對象的方式不一樣,前者是使用類加載機制,後者是建立一個新類。

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

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

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

factory = (ExampleInterface)c.newInstance();

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

String className = “Example”;

class c = Class.forName(className);

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

String className = readfromXMlConfig;//從xml 配置檔案中獲得字元串

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

從JVM的角度看,我們使用關鍵字new建立一個類的時候,這個類可以沒有被加載。但是使用newInstance()方法的時候,就必須保證:

1、這個類已經加載;

2、這個類已經連接配接了。

而完成上面兩個步驟的正是Class的靜态方法forName()所完成的,這個靜态方法調用了啟動類加載器,即加載 java API的那個加載器。

現在可以看出,newInstance()實際上是把new這個方式分解為兩步,即首先調用Class加載方法加載某個類,然後執行個體化。

這樣分步的好處是顯而易見的。我們可以在調用class的靜态加載方法forName時獲得更好的靈活性,提供給了一種降耦的手段。

二.new 和Class.forName()有什麼差別?

其實上面已經說到一些了,這裡來做個總結:

首先,newInstance( )是一個方法,而new是一個關鍵字;

其次,Class下的newInstance()的使用有局限,因為它生成對象隻能調用無參的構造函數,而使用 new關鍵字生成對象沒有這個限制。

簡言之:

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

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

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

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

三.為什麼在加載資料庫驅動包的時候有用的是Class.forName( ),卻沒有調用newInstance( )?

在Java開發特别是資料庫開發中,經常會用到Class.forName( )這個方法。

通過查詢Java Documentation我們會發現使用Class.forName( )靜态方法的目的是為了動态加載類。

通常編碼過程中,在加載完成後,一般還要調用Class下的newInstance( )靜态方法來執行個體化對象以便操作。是以,單使用Class.forName( )是動态加載類是沒有用的,其最終目的是為了執行個體化對象。

相關英文參考文獻如下:
we just want to load the driver to jvm only, but not need to user the instance of driver,
so call Class.forName(xxx.xx.xx) is enough, if you call Class.forName(xxx.xx.xx).newInstance(),
the result will same as calling Class.forName(xxx.xx.xx),
because Class.forName(xxx.xx.xx).newInstance() will load driver first,
and then create instance, but the instacne you will never use in usual,
so you need not to create it.

========================== 華麗的分割線 =============================

      

Class.forName()、Class.forName().newInstance() 、New 三者差別!

終于明白為什麼加載資料庫驅動隻用Class.forName()了!!困擾了我2個小時!!希望我寫的這個東西對各位有所幫助。

   在Java開發特别是資料庫開發中,經常會用到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);就可以了。

貼出Proxool 連接配接池的靜态初始化方法:

public class ProxoolDriver implements Driver {

    private static final Log LOG = LogFactory.getLog(ProxoolDriver.class);

    static {

        try {

            DriverManager.registerDriver(new ProxoolDriver());

        } catch (SQLException e) {

            System.out.println(e.toString());

        }

    }

}