轉自:https://www.cnblogs.com/ljbguanli/p/7107058.html
Java程序在執行時,Java執行時系統一直對全部的對象進行所謂的執行時類型標識。這項信息紀錄了每一個對象所屬的類。虛擬機通常使用執行時類型信息選擇正確方法去執行,用來保存這些運行時類型信息的類是Class類。Class類封裝一個對象和接口執行時的狀態,當裝載類時。Class類型的對象自己主動創建。
Class 沒有公共構造方法,因此不能顯式地聲明一個Class對象,Class 對象是在載入類時由Java 虛擬機以及通過調用類載入器中的 defineClass 方法自己主動構造的。
虛擬機為每種類型管理一個獨一無二的Class對象。也就是說,每一個類(型)都有一個Class對象。執行程序時。Java虛擬機(JVM)首先檢查是否所要載入的類相應的Class對象是否已經載入。假設沒有載入,JVM就會依據類名查找.class文件,並將其Class對象載入。
主要的 Java 類型(boolean、byte、char、short、int、long、float 和 double)和keyword void 也都相應一個 Class 對象。
每一個數組屬於被映射為 Class 對象的一個類,全部具有同樣元素類型和維數的數組都共享該 Class 對象。
一般某個類的Class對象被加載內存,它就用來創建這個類的全部對象。
一、怎樣得到Class的對象呢?有三種方法能夠的獲取:
1、調用Object類的getClass()方法來得到Class對象,這也是最常見的產生Class對象的方法。比如:
MyObject x;
Class c1 = x.getClass();
2、使用Class類的中靜態forName()方法獲得與字符串相應的Class對象。比如:
Class c2=Class.forName("MyObject"),Employee必須是接口或者類的名字。
3、獲取Class類型對象的第三個方法很easy。假設T是一個Java類型。那么T.class就代表了匹配的類對象。
比如
Class cl1 = Manager.class;
Class cl2 = int.class;
Class cl3 = Double[].class;
注意:Class對象實際上描寫敘述的僅僅是類型。而這類型未必是類或者接口。
比如上面的int.class是一個Class類型的對象。
因為歷史原因。數組類型的getName方法會返回奇怪的名字。
二、Class類的經常用法
1、getName()
一個Class對象描寫敘述了一個特定類的屬性,Class類中最經常使用的方法getName以 String 的形式返回此 Class 對象所表示的實體(類、接口、數組類、基本類型或 void)名稱。
2、newInstance()
Class另一個實用的方法能夠為類創建一個實例,這種方法叫做newInstance()。比如:
x.getClass.newInstance(),創建了一個同x一樣類型的新實例。newInstance()方法調用默認構造器(無參數構造器)初始化新建對象。
3、getClassLoader()
返回該類的類載入器。
4、getComponentType()
返回表示數組組件類型的 Class。
5、getSuperclass()
返回表示此 Class 所表示的實體(類、接口、基本類型或 void)的超類的 Class。
6、isArray()
判定此 Class 對象是否表示一個數組類。
三、Class的一些使用技巧
1、forName和newInstance結合起來使用,能夠依據存儲在字符串中的類名創建對象。比如
Object obj = Class.forName(s).newInstance();
2、虛擬機為每種類型管理一個獨一無二的Class對象。因此能夠使用==操作符來比較類對象。比如:
if(e.getClass() == Employee.class)...
2、 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的值為當前類的類載入器。
|
forName Returns the |
|
forName Returns the |
說明:
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都要裝載在虛擬機上才干執行。
1. forName這句話就是裝載類用的(new是依據載入到內存中的類創建一個實例,要分清楚)。
2. 至於什么時候用。能夠考慮一下這個問題,給你一個字符串變量,它代表一個類的包名和類名,你怎么實例化它?
A a = (A)Class.forName("pacage.A").newInstance();這和 A a =new A();是一樣的效果。
3. jvm在裝載類時會運行類的靜態代碼段,要記住靜態代碼是和class綁定的,class裝載成功就表示運行了你的靜態代碼了,並且以后不會再運行這段靜態代碼了。
4. Class.forName(xxx.xx.xx)的作用是要求JVM查找並載入指定的類,也就是說JVM會運行該類的靜態代碼段。
5. 動態載入和創建Class 對象,比方想依據用戶輸入的字符串來創建對象
String str = 用戶輸入的字符串
Class t = Class.forName(str);
t.newInstance();
二.在初始化一個類,生成一個實例的時候。newInstance()方法和newkeyword除了一個是方法,一個是keyword外,最主要有什么差別?
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的角度看,我們使用keywordnew創建一個類的時候。這個類能夠沒有被載入。 可是使用newInstance()方法的時候。
就必須保證:
1、這個類已經載入;
2、這個類已經連接了。
而完畢上面兩個步驟的正是Class的靜態方法forName()所完畢的。這個靜態方法調用了啟動類載入器,即載入 java API的那個載入器。
如今能夠看出。newInstance()實際上是把new這個方式分解為兩步。即首先調用Class載入方法載入某個類,然后實例化。這樣分步的優點是顯而易見的。我們能夠在調用class的靜態載入方法forName時獲得更好
的靈活性,提供給了一種降耦的手段。
三.最后用最簡單的描寫敘述來區分newkeyword和newInstance()方法的差別:
1. newInstance: 弱類型。低效率。僅僅能調用無參構造。
2. new: 強類型。相對高效。
能調用不論什么public構造。
3、應用情景:
情景一:載入數據庫驅動的時候
Class.forName的一個非經常見的使用方法是在載入數據庫驅動的時候。
如:
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();
參考資料:
JDK1.8_API.../docs/api/java/lang/Class.html
http://www.ibm.com/developerworks/cn/java/j-lo-classloader/