一、Class類簡介:
Java程 序在運行時,Java運行時系統一直對全部的對象進行所謂的運行時類型標識。這項信息紀錄了每一個對象所屬的類。虛擬機一般使用運行時類型信息選準正確方法 去執行,用來保存這些類型信息的類是Class類。Class類封裝一個對象和接口運行時的狀態,當裝載類時,Class類型的對象自動建立。
Class 沒有公共構造方法。Class 對象是在加載類時由Java 虛擬機以及經過調用類加載器中的 defineClass 方法自動構造的,所以不能顯式地聲明一個Class對象。
虛擬機爲每種類型管理一個獨一無二的Class對象。也就是說,每一個類(型)都有一個Class對象。運行程序時,Java虛擬機(JVM)首先檢查是否 所要加載的類對應的Class對象是否已經加載。若是沒有加載,JVM就會根據類名查找.class文件,並將其Class對象載入。
基本的 Java 類型(boolean、byte、char、short、int、long、float 和 double)和關鍵字 void 也都對應一個 Class 對象。
每一個數組屬於被映射爲 Class 對象的一個類,全部具備相同元素類型和維數的數組都共享該 Class 對象。
通常某個類的Class對象被載入內存,它就用來建立這個類的全部對象。java
1、如何獲得Class的對象呢?有三種方法能夠的獲取:web
一、調用Object類的getClass()方法來獲得Class對象,這也是最多見的產生Class對象的方法。例如:
MyObject x;
Class c1 = x.getClass();sql
二、使用Class類的中靜態forName()方法得到與字符串對應的Class對象。例如:
Class c2=Class.forName(「MyObject」),Employee必須是接口或者類的名字。數據庫
三、獲取Class類型對象的第三個方法很是簡單。若是T是一個Java類型,那麼T.class就表明了匹配的類對象。例如
Class cl1 = Manager.class;
Class cl2 = int.class;
Class cl3 = Double[].class;
注意:Class對象實際上描述的只是類型,而這類型未必是類或者接口。例如上面的int.class是一個Class類型的對象。因爲歷史緣由,數組類型的getName方法會返回奇怪的名字。數組
2、Class類的經常使用方法jvm
一、getName()svg
一個Class對象描述了一個特定類的屬性,Class類中最經常使用的方法getName以 String 的形式返回此 Class 對象所表示的實體(類、接口、數組類、基本類型或 void)名稱。sqlserver
二、newInstance()ui
Class還有一個有用的方法能夠爲類建立一個實例,這個方法叫作newInstance()。例如:
x.getClass.newInstance(),建立了一個同x同樣類型的新實例。newInstance()方法調用默認構造器(無參數構造器)初始化新建對象。this
三、getClassLoader()
返回該類的類加載器。
四、getComponentType()
返回表示數組組件類型的 Class。
五、getSuperclass()
返回表示此 Class 所表示的實體(類、接口、基本類型或 void)的超類的 Class。
六、isArray()
斷定此 Class 對象是否表示一個數組類。
3、Class的一些使用技巧
一、forName和newInstance結合起來使用,能夠根據存儲在字符串中的類名建立對象。例如
Object obj = Class.forName(s).newInstance();
二、虛擬機爲每種類型管理一個獨一無二的Class對象。所以可使用==操做符來比較類對象。例如:
if(e.getClass() == Employee.class)…
二、 Class.forName()方法:
Class.forName:返回與給定的字符串名稱相關聯類或接口的Class對象。
Class.forName是一個靜 態方法,一樣能夠用來加載類。該方法有兩種形式:Class.forName(String name, boolean initialize, ClassLoader loader)和 Class.forName(String className)。第一種形式的參數 name表示的是類的全名;initialize表示是否初始化類;loader表示加載時使用的類加載器。第 二種形式則至關於設置了參數 initialize的值爲 true,loader的值爲當前類的類加載器。
static Class<?>
forName(String className)
Returns the Class object associated with the class or interface with the given string name.
static Class<?>
forName(String name, boolean initialize, ClassLoader loader)
Returns the Class object associated with the class or interface with the given string name, using the given class loader.
說明:
publicstatic Class<?> forName(String className)
Returns the Class object associated withthe class or interface with the given string name. Invokingthis method is equivalent to:
Class.forName(className,true, currentLoader)
where currentLoader denotes the definingclass loader of the current class.
For example, thefollowing code fragment returns the runtime Class descriptor for theclass named java.lang.Thread:
Class t =Class.forName(「java.lang.Thread」)
A call to forName(「X」) causes theclass named X to beinitialized.
Parameters:
className - the fully qualifiedname of the desired class.
Returns:
the Class object for the classwith the specified name.
從官方給出的API文檔中能夠看出:
Class.forName(className)其實是調用Class.forName(className,true, this.getClass().getClassLoader())。第二個參數,是指Class被loading後是否是必須被初始化。能夠看出,使 用Class.forName(className)加載類時則已初始化。
因此Class.forName(className)能夠簡單的理解爲:得到字符串參數中指定的類,並初始化該類。
一.首先你要明白在java裏面任何class都要裝載在虛擬機上才能運行。
forName這句話就是裝載類用的(new是根據加載到內存中的類建立一個實例,要分清楚)。
至於何時用,能夠考慮一下這個問題,給你一個字符串變量,它表明一個類的包名和類名,你怎麼實例化它? A a = (A)Class.forName("pacage.A").newInstance();這和 A a =new A();是同樣的效果。
jvm在裝載類時會執行類的靜態代碼段,要記住靜態代碼是和class綁定的,class裝載成功就表示執行了你的靜態代碼了,並且之後不會再執行這段靜態代碼了。
Class.forName(xxx.xx.xx)的做用是要求JVM查找並加載指定的類,也就是說JVM會執行該類的靜態代碼段。
動態加載和建立Class 對象,好比想根據用戶輸入的字符串來建立對象 String str = 用戶輸入的字符串 Class t = Class.forName(str); t.newInstance();
二.在初始化一個類,生成一個實例的時候,newInstance()方法和new關鍵字除了一個是方法,一個是關鍵字外,最主要有什麼區別?
1.它們的區別在於建立對象的方式不同,前者是使用類加載機制,後者是建立一個新類。 2.那麼爲何會有兩種建立對象方式? 這主要考慮到軟件的可伸縮、可擴展和可重用等軟件設計思想。 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就能夠。 3.從JVM的角度看,咱們使用關鍵字new建立一個類的時候,這個類能夠沒有被加載。 可是使用newInstance()方法的時候, 就必須保證: 一、這個類已經加載; 二、這個類已經鏈接了。 而完成上面兩個步驟的正是Class的靜態方法forName()所完成的,這個靜態方法調用了啓動類加載器,即加載 java API的那個加載器。 如今能夠看出,newInstance()其實是把new這個方式分解爲兩步,即首先調用Class加載方法加載某個類,而後實例化。這樣分步的好處是顯而易見的。咱們能夠在調用class的靜態加載方法forName時得到更好 的靈活性,提供給了一種降耦的手段。
三.最後用最簡單的描述來區分new關鍵字和newInstance()方法的區別:
三、應用情景:
情景一:加載數據庫驅動的時候
Class.forName的一個很常見的用法是在加載數據庫驅動的時候。
如:
[java] view plain copy
Class.forName(「com.microsoft.sqlserver.jdbc.SQLServerDriver」);
Connection con=DriverManager.getConnection(「jdbc:sqlserver://localhost:1433;DatabaseName==JSP」,「jph」,「jph」);
爲何在咱們加載數據庫驅動包的時候有的卻沒有調用newInstance( )方法呢?
即有的jdbc鏈接數據庫的寫法裏是Class.forName(xxx.xx.xx);而有一些:Class.forName(xxx.xx.xx).newInstance(),爲何會有這兩種寫法呢?
剛纔提到,Class.forName("");的做用是要求JVM查找並加載指定的類,若是在類中有靜態初始化器的話,JVM必然會執行該類的靜態代碼段。
而在JDBC規範中明確要求這個Driver類必須向DriverManager註冊本身,即任何一個JDBCDriver的Driver類的代碼都必須相似以下:
public classMyJDBCDriver implements Driver {
static{ DriverManager.registerDriver(new MyJDBCDriver());
}
}
既然在靜態初始化器的中已經進行了註冊,因此咱們在使用JDBC時只須要Class.forName(XXX.XXX);就能夠了。
情景二:使用AIDL與電話管理Servic進行通訊
Method method =Class.forName(「Android.os.ServiceManager」)
.getMethod("getService",String.class);
// 獲取遠程TELEPHONY_SERVICE的IBinder對象的代理
IBinder binder =(IBinder) method.invoke(null, new Object[] { TELEPHONY_SERVICE});
// 將IBinder對象的代理轉換爲ITelephony對象
ITelephonytelephony = ITelephony.Stub.asInterface(binder);
// 掛斷電話
telephony.endCall();