java中反射的三種方法:
1. 通過Object類的getClass方法來獲取
java.lang.Object中定義有g(shù)etClass方法:public final Class getClass()
所有Java對象都具備這個方法,該方法用于返回調(diào)用該方法的對象的所屬類關(guān)聯(lián)的Class對象,例如:
Date date1 = new Date();
Date date2 = new Date();
Class c1 = date1.getClass();
Class c2 = date2.getClass();
System.out.println(c1.getName());
// java.util.Date
System.out.println(c1 == c2);
// true
上面的代碼中,調(diào)用Date對象date1的getClass方法將返回用于封裝Date類信息的Class對象。
這里調(diào)用了Class類的getName方法:public String getName(),這個方法的含義很直觀,即返回所封裝的類的名稱。
需要注意的是,代碼中的date1和date2的getClass方法返回了相同的Class對象(c1==c2的值為true)。這是因為,對于相同的類,JVM只會載入一次,而與該類對應(yīng)的Class對象也只會存在一個,無論該類實例化了多少對象。
另外,需要強調(diào)的是,當(dāng)一個對象被其父類的引用或其實現(xiàn)的接口類型的引用所指向時,getClass方法返回的是與對象實際所屬類關(guān)聯(lián)的Class對象。例如:
List list = new ArrayList();
System.out.println(list.getClass().getName()); // java.util.ArrayList
上面的代碼中,語句list.getClass()方法返回的是list所指向?qū)ο髮嶋H所屬類java.util.ArrayList對應(yīng)的 Class對象而并未java.util.List所對應(yīng)的Class對象。有些時候可以通過這個方法了解一個對象的運行時類型,例如:
HashSet set = new HashSet();
Iterator it = set.iterator();
System.out.println(it.getClass().getName()); //java.util.HashMap$KeyIterator
從代碼可以看出,HashSet的iterator方法返回的是實現(xiàn)了Iterator接口的HashMap內(nèi)部類(KeyIterator)對象。
因為抽象類和接口不可能實例化對象,因此不能通過Object的getClass方法獲得與抽象類和接口關(guān)聯(lián)的Class對象。
2. 使用.class的方式
使用類名加“.class”的方式即會返回與該類對應(yīng)的Class對象。例如:
Class clazz = String.class;
System.out.println(clazz.getName()); // java.lang.String
這個方法可以直接獲得與指定類關(guān)聯(lián)的Class對象,而并不需要有該類的對象存在。
3. 使用Class.forName方法
Class有一個著名的static方法forName:public static Class forName(String className) throws ClassNotFoundException
該方法可以根據(jù)字符串參數(shù)所指定的類名獲取與該類關(guān)聯(lián)的Class對象。如果該類還沒有被裝入,該方法會將該類裝入JVM。
該方法聲明拋出ClassNotFoundException異常。顧名思義,當(dāng)該方法無法獲取需要裝入的類時(例如,在當(dāng)前類路徑中不存在這個類),就會拋出這個異常。
例如,如果當(dāng)前類路徑中存在Foo類:
package org.whatisjava.reflect;
public class Foo {
public Foo() {
System.out.println("Foo()");
}
static {
System.out.println("Foo is initialized");
}
}
運行下面的代碼:
Class clazz = Class.forName("org.whatisjava.reflect.Foo");
控制臺會有如下輸出:
Foo is initialized
Class.forName("org.whatisjava.reflect.Foo")首先會將reflection.Foo類裝入JVM,并返回與之關(guān)聯(lián)的Class對象。JVM裝入Foo類后對其進行初始化,調(diào)用了其static塊中的代碼。需要注意的是:forName方法的參數(shù)是類的完 整限定名(即包含包名)。
區(qū)別于前面兩種獲取Class對象的方法:使用Class.forName方法所要獲取的與之對應(yīng)的Class對象的類可以通過字符串的方式給定。該方法通常用于在程序運行時根據(jù)類名動態(tài)的載入該類并獲得與之對應(yīng)的Class對象。
通過上面的文章相信你對java的反射機制有了一定的認識,同時也對java中Class類的用法有了比較清晰的理解,在我們實際工作的過程中,我們不斷的運用java知識來解決實際生活中的問題的時候我們就能對java反射機制有一個更深入的理解!
二、代碼示例
1.ClassTest.java
[java] view plaincopy
/**
* java中Class類的使用
*/
import java.io.*;
import java.lang.reflect.*;
public class ClassTest1 {
public ClassTest1(){
}
public static void main(String[] args) throws Exception{
ClassTest1 test=new ClassTest1();
ClassTest1 test1=test.getClass().newInstance();
//test1=test;
test.printMessage();
test1.printMessage();
System.out.println(test.hashCode());
System.out.println(test1.hashCode());
Method[] method=test1.getClass().getMethods();
for(Method m :method){
System.out.println(m.getDeclaringClass());
System.out.println(m.getName());
}
}
public void printMessage(){
System.out.println("Created successful!");
}
}
運行結(jié)果:
[plain] view plaincopy
Created successful!
Created successful!
14576877
12677476
class ClassTest1
printMessage
class ClassTest1
main
class java.lang.Object
wait
class java.lang.Object
wait
class java.lang.Object
wait
class java.lang.Object
hashCode
class java.lang.Object
getClass
class java.lang.Object
equals
class java.lang.Object
toString
class java.lang.Object
notify
class java.lang.Object
notifyAll
2.TestClass.java
[java] view plaincopy
/**
*
*/
public class TestClass {
public static void main(String[] args)
{
try {
// 測試Class.forName()
Class testTypeForName = Class.forName("TestClassType");
System.out.println("testForName---" + testTypeForName);
// 測試類名.class
Class testTypeClass = TestClassType.class;
System.out.println("testTypeClass---" + testTypeClass);
// 測試Object.getClass()
TestClassType testGetClass = new TestClassType();
System.out.println("testGetClass---" + testGetClass.getClass());
} catch (ClassNotFoundException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
class TestClassType {
// 構(gòu)造函數(shù)
public TestClassType() {
System.out.println("----構(gòu)造函數(shù)---");
}
// 靜態(tài)的參數(shù)初始化
static {
System.out.println("---靜態(tài)的參數(shù)初始化---");
}
// 非靜態(tài)的參數(shù)初始化
{
System.out.println("----非靜態(tài)的參數(shù)初始化---");
}
}
運行結(jié)果:
[plain] view plaincopy
---靜態(tài)的參數(shù)初始化---
testForName---class TestClassType
testTypeClass---class TestClassType
----非靜態(tài)的參數(shù)初始化---
----構(gòu)造函數(shù)---
testGetClass---class TestClassType
分析:根據(jù)結(jié)果可以發(fā)現(xiàn),三種生成的Class對象一樣的,并且三種生成Class對象只打印一次“靜態(tài)的參數(shù)初始化”。
作者:傳智播客JavaEE培訓(xùn)學(xué)院
首發(fā):http://java.itcast.cn