Java之Class.forName()用法案例詳解
Class.forName()主要功能
Class.forName(xxx.xx.xx)返回的是一個類,
Class.forName(xxx.xx.xx)的作用是要求JVM查找并加載指定的類,也就是說JVM會執(zhí)行該類的靜態(tài)代碼段。
下面,通過解答以下三個問題的來詳細講解下Class.forName()的用法。
一.什么時候用Class.forName()?
給你一個字符串變量,它代表一個類的包名和類名,你怎么實例化它?你第一想到的肯定是new,但是注意一點:
A a = (A)Class.forName(“pacage.A”).newInstance();
這和你 A a = new A(); 是一樣的效果。
現(xiàn)在言歸正傳。
動態(tài)加載和創(chuàng)建Class 對象,比如想根據(jù)用戶輸入的字符串來創(chuàng)建對象時需要用到:
String str = “用戶輸入的字符串” ; Class t = Class.forName(str); t.newInstance();
在初始化一個類,生成一個實例的時候,newInstance()方法和new關(guān)鍵字除了一個是方法,一個是關(guān)鍵字外,最主要有什么區(qū)別?它們的區(qū)別在于創(chuàng)建對象的方式不一樣,前者是使用類加載機制,后者是創(chuàng)建一個新類。那么為什么會有兩種創(chuàng)建對象方式?這主要考慮到軟件的可伸縮、可擴展和可重用等軟件設(shè)計思想。
Java中工廠模式經(jīng)常使用newInstance()方法來創(chuàng)建對象,因此從為什么要使用工廠模式上可以找到具體答案。 例如:
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();
上面代碼已經(jīng)不存在Example的類名稱,它的優(yōu)點是,無論Example類怎么變化,上述代碼不變,甚至可以更換Example的兄弟類Example2 , Example3 , Example4……,只要他們繼承ExampleInterface就可以。
從JVM的角度看
我們使用關(guān)鍵字new創(chuàng)建一個類的時候,這個類可以沒有被加載。但是使用newInstance()方法的時候,就必須保證:
1、這個類已經(jīng)加載;
2、這個類已經(jīng)連接了。
而完成上面兩個步驟的正是Class的靜態(tài)方法forName()所完成的,這個靜態(tài)方法調(diào)用了啟動類加載器,即加載 java API的那個加載器。
現(xiàn)在可以看出,newInstance()實際上是把new這個方式分解為兩步,即首先調(diào)用Class加載方法加載某個類,然后實例化。 這樣分步的好處是顯而易見的。我們可以在調(diào)用class的靜態(tài)加載方法forName時獲得更好的靈活性,提供給了一種降耦的手段。
二.new 和Class.forName()有什么區(qū)別?
其實上面已經(jīng)說到一些了,這里來做個總結(jié):
首先,newInstance( )是一個方法,而new是一個關(guān)鍵字;
其次,Class下的newInstance()的使用有局限,因為它生成對象只能調(diào)用無參的構(gòu)造函數(shù),而使用 new關(guān)鍵字生成對象沒有這個限制。
簡言之:
newInstance(): 弱類型,低效率,只能調(diào)用無參構(gòu)造。
new: 強類型,相對高效,能調(diào)用任何public構(gòu)造。
Class.forName(“”)返回的是類。
Class.forName(“”).newInstance()返回的是object 。
三.為什么在加載數(shù)據(jù)庫驅(qū)動包的時候有用的是Class.forName( ),卻沒有調(diào)用newInstance( )?
在Java開發(fā)特別是數(shù)據(jù)庫開發(fā)中,經(jīng)常會用到Class.forName( )這個方法。
通過查詢Java Documentation我們會發(fā)現(xiàn)使用Class.forName( )靜態(tài)方法的目的是為了動態(tài)加載類。
通常編碼過程中,在加載完成后,一般還要調(diào)用Class下的newInstance( )靜態(tài)方法來實例化對象以便操作。因此,單單使用Class.forName( )動態(tài)加載類是沒有用的,其最終目的是為了實例化對象。
有數(shù)據(jù)庫開發(fā)經(jīng)驗朋友會發(fā)現(xiàn),為什么在我們加載數(shù)據(jù)庫驅(qū)動包的時候有的卻沒有調(diào)用newInstance( )方法呢?
即有的jdbc連接數(shù)據(jù)庫的寫法里是Class.forName(xxx.xx.xx);而有一 些:Class.forName(xxx.xx.xx).newInstance(),為什么會有這兩種寫法呢?
剛才提到,Class.forName(“”);的作用是要求JVM查找并加載指定的類,首先要明白,java里面任何class都要裝載在虛擬機上才能運行,而靜態(tài)代碼是和class綁定的,class裝載成功就表示執(zhí)行了你的靜態(tài)代碼了,而且以后不會再走這段靜態(tài)代碼了。
而我們前面也說了,Class.forName(xxx.xx.xx)的作用就是要求JVM查找并加載指定的類,如果在類中有靜態(tài)初始化器的話,JVM必然會執(zhí)行該類的靜態(tài)代碼段。
而在JDBC規(guī)范中明確要求這個Driver類必須向DriverManager注冊自己,即任何一個JDBC Driver的 Driver類的代碼都必須類似如下:
public class MyJDBCDriver implements Driver {
static {
DriverManager.registerDriver(new MyJDBCDriver());
}
}
既然在靜態(tài)初始化器的中已經(jīng)進行了注冊,所以我們在使用JDBC時只需要Class.forName(XXX.XXX);就可以了。
到此這篇關(guān)于Java之Class.forName()用法案例詳解的文章就介紹到這了,更多相關(guān)Java之Class.forName()用法內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Mybatis-plus實現(xiàn)主鍵自增和自動注入時間的示例代碼
這篇文章主要介紹了Mybatis-plus實現(xiàn)主鍵自增和自動注入時間的示例代碼,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-07-07
java新增關(guān)聯(lián)的三張表,每張表要求都插入集合,代碼實現(xiàn)方式
這篇文章主要介紹了java新增關(guān)聯(lián)的三張表,每張表要求都插入集合,代碼實現(xiàn)方式,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2023-12-12
springboot如何根據(jù)不同的日志級別顯示不同的顏色
這篇文章主要介紹了springboot如何根據(jù)不同的日志級別顯示不同的顏色問題,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2023-08-08
Spring security用戶URL權(quán)限FilterSecurityInterceptor使用解析
這篇文章主要介紹了Spring security用戶URL權(quán)限FilterSecurityInterceptor使用解析,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下2019-12-12

