ClassLoader.loadClass和Class.forName的區別


為什么要把ClassLoader.loadClass(String name)和Class.forName(String name)進行比較呢,因為他們都能在運行時對任意一個類,都能夠知道該類的所有屬性和方法;對於任意一個對象,都能夠調用它的任意方法和屬性。

在比較它倆之前需先了解一下java類裝載的過程

java類裝載過程分為3步:

   

 

  1:加載

    Jvm把class文件字節碼加載到內存中,並將這些靜態數據裝換成運行時數據區中方法區的類型數據,在運行時數據區堆中生成一個代表這個類

  的java.lang.Class對象,作為方法區類數據的訪問入口。

  *釋:方法區不僅僅是存放方法,它存放的是類的類型信息。

  2:鏈接:執行下面的校驗、准備和解析步驟,其中解析步驟是可選的

    a:校驗:檢查加載的class文件的正確性和安全性

    b:准備:為類變量分配存儲空間並設置類變量初始值,類變量隨類型信息存放在方法區中,生命周期很長,使用不當和容易造成內存泄漏。

    *:類變量就是static變量;初始值指的是類變量類型的默認值而不是實際要賦的值

    c:解析:jvm將常量池內的符號引用轉換為直接引用

  3:初始化:執行類變量賦值和靜態代碼塊

 

在了解了類裝載過程之后我們繼續比較二者區別:

  • Classloder.loaderClass(String name)

    其實該方法內部調用的是:Classloder. loadClass(name, false)

    方法:Classloder. loadClass(String name, boolean resolve)

        1:參數name代表類的全限定類名

        2:參數resolve代表是否解析,resolve為true是解析該類

  • Class.forName(String name)

    其實該方法內部調用的是:Class.forName(className, true, ClassLoader.getClassLoader(caller))

    方法:Class.forName0(String name, boolean initialize, ClassLoader loader)

      參數name代表全限定類名

      參數initialize表示是否初始化該類,為true是初始化該類

      參數loader 對應的類加載器

  • 兩者最大的區別

    Class.forName得到的class是已經初始化完成的

    Classloder.loaderClass得到的class是還沒有鏈接的

  • 怎么使用

    有些情況是只需要知道這個類的存在而不需要初始化的情況使用Classloder.loaderClass,而有些時候又必須執行初始化就選擇Class.forName

    例如:數據庫驅動加載就是使用Class.froName(“com.mysql.jdbc.Driver”),

    下面我們來看看Driver的源代碼:

public class Driver extends NonRegisteringDriver implements java.sql.Driver {

    public Driver() throws SQLException {

    } 
static { try { DriverManager.registerDriver(new Driver()); } catch (SQLException var1) { throw new RuntimeException("Can\'t register driver!"); } } }

    從Driver的源碼中我們可以看出Driver這個類只有一個static塊,這樣我們需要初始化后才能得到DriverManager,所以我們選擇使用Class.forName()

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM