java反射是如何實現(xiàn)的?這篇文章運用了實例代碼展示,代碼非常詳細(xì),可供感興趣的小伙伴們參考借鑒,希望對大家有所幫助。
網(wǎng)站建設(shè)哪家好,找創(chuàng)新互聯(lián)公司!專注于網(wǎng)頁設(shè)計、網(wǎng)站建設(shè)、微信開發(fā)、小程序設(shè)計、集團企業(yè)網(wǎng)站建設(shè)等服務(wù)項目。為回饋新老客戶創(chuàng)新互聯(lián)還提供了李滄免費建站歡迎大家使用!Java反射
在 Java 中,并不是所有的類型信息都能在編譯階段明確,有一些類型信息需要在運行時才能確定,這種機制被稱為 RTTI,即運行時類型識別,運行時類型識別主要由Class類實現(xiàn)。
1、 Class類
在Java中,我們常用“class”(首字母為小寫的c)關(guān)鍵字來定義一個類,說這個類是對某一類對象的抽象。你比如說王二是一個網(wǎng)絡(luò)知名作者,我們可以這樣簡單地定義作者類:
package com.cmower.java_demo.fifteen;
class Author {
private String pen_name;
private String real_name;}
現(xiàn)在,我們想知道Writer這個類本身的一些信息(比如說類名),該怎么辦呢?這時候就需要用到“Class”(首字母為大寫的C)類,該類包含了與類有關(guān)的信息。請看以下代碼:
public class Test {
public static void main (String [] args) {
Author wanger = new Author();
Class c1 = wanger.getClass();
System.out.println(c1.getName());
//輸出 com.cmower.java_demo.fifteen.Author
}}
當(dāng)我們創(chuàng)建了作者對象wanger后,就可以通過wanger.getClass()獲取wanger的Class對象,通過c1.getName()可獲得wanger對象的類名。
想象一下,經(jīng)過五年的刻意練習(xí),張三從一名寫作愛好者晉升為一名作家了。我們用代碼來假裝一下:
package com.cmower.java_demo.fifteen;
class Author {
private String pen_name;
private String real_name;}
class Writer extends Author {
private String honour;}
public class Test {
public static void main (String [] args) {
Author wanger = new Writer();
Class c1 = wanger.getClass();
System.out.println(c1.getName());
//輸出 com.cmower.java_demo.fifteen.Writer
}}
在上例中,即使我們將Writer的對象引用wanger向上轉(zhuǎn)型為Author,wanger的Class對象類型依然是Writer(通過輸出結(jié)果可以判定)。這也就是說,Java能夠在運行時自動識別類型的信息,它不會因為wanger的引用類型是Author而丟失wanger真正的類型信息(Writer)。Java是怎么做到這一點呢?
當(dāng)Java創(chuàng)建某個類的對象,比如Writer類對象時,Java會檢查內(nèi)存中是否有相應(yīng)的Class對象。如果內(nèi)存中沒有相應(yīng)的Class對象,那么Java會在.class文件中尋找Writer類的定義,并加載Writer類的Class對象。
一旦Class對象加載成功,就可以用它來創(chuàng)建這種類型的所有對象。這也就是說,每個對象在運行時都會有對應(yīng)的Class對象,這個Class對象包含了這個對象的類型信息。因此,我們能夠通過Class對象知道某個對象“真正”的類型,并不會因為向上轉(zhuǎn)型而丟失。
02、 獲取Class對象的其他方式
在使用getClass()方法獲取一個類的Class對象時,我們必須要先獲取這個類的對象,比如上面提到的wanger。如果我們之前沒有獲取這個類的對象,就需要用另外兩種方式來獲取類的Class對象:
Class c2 = Writer.class;System.out.println(c2.getName());
try {
Class c3 = Class.forName("com.cmower.java_demo.fifteen.Writer");
System.out.println(c3.getName());} catch (ClassNotFoundException e) {
e.printStackTrace();}
1)當(dāng)使用.class來獲取Class對象時,不會自動地初始化該Class對象,初始化被延遲到了對靜態(tài)方法或者非final靜態(tài)域進行首次引用時才執(zhí)行。這樣做不僅更簡單,而且更安全,因為它在編譯時就會受到檢查(因此不需要置于try語句塊中)。
2)Class.forName會自動地初始化該Class對象,但需要指定類名,并且需要置于try語句塊中。
03、 Class類提供的常用方法
Class類為我們提供了一些非常有用的方法,比如說getName()用來返回類名,getPackage()返回類所在的包名。
我們還可以利用Class類提供的newInstance()方法來創(chuàng)建相應(yīng)類的對象,比如:
Class c2 = Writer.class;System.out.println(c2.getName());
try {
Writer wangsan = (Writer) c2.newInstance();
System.out.println(wangsan);
// 輸出:com.cmower.java_demo.fifteen.Writer@7852e922} catch (InstantiationException | IllegalAccessException e1) {
e1.printStackTrace();}
由于我們在創(chuàng)建Class對象c2時沒有使用泛型,所以newInstance()返回的對象類型需要強轉(zhuǎn)為Writer。我們可以在此基礎(chǔ)上進行改進,示例如下:
Class
try {
Writer wangsan = c4.newInstance();
System.out.println(wangsan);
// 輸出:com.cmower.java_demo.fifteen.Writer@7852e922} catch (InstantiationException | IllegalAccessException e1) {
e1.printStackTrace();}
04、 反射
我們還可以通過getFields()獲取所有public修飾的字段,通過getMethods()返回所有public修飾的方法。
甚至,我們還可以通過getDeclaredFields()獲取更多字段,包括公共、受保護、默認(rèn)(包)訪問和私有字段,但不包括繼承字段。對應(yīng)的,getDeclaredMethods()用來獲取更多方法。示例如下:
package com.cmower.java_demo.fifteen;
import java.lang.reflect.Field;import java.lang.reflect.Method;
class Author {
private String pen_name;
private String real_name;}
class Writer extends Author {
private String honour;
private void makeMoney() {
System.out.println("很多很多錢");
}}
public class Test {
public static void main(String[] args) {
Class c4 = Writer.class;
System.out.println(c4.getName());
try {
Writer wangsan = c4.newInstance();
System.out.println(wangsan);
Field[] fields = c4.getDeclaredFields();
for (Field field : fields) {
System.out.println(field.getName());
}
Method[] methods = c4.getDeclaredMethods();
for (Method method : methods) {
System.out.println(method.getName());
}
} catch (InstantiationException | IllegalAccessException e1) {
e1.printStackTrace();
}
}}
上面的例子其實涉及到了反射,F(xiàn)ield、Method(還有例子中未提到的Constructor)都來自java.lang.reflect類庫。Class類與java.lang.reflect類庫一起對反射的概念進行了支持。
有時候,我們需要從磁盤文件或網(wǎng)絡(luò)文件中讀取一串字節(jié)碼,并把它轉(zhuǎn)換成一個類,這時候就需要用到反射。最常見的典型例子就是將一串JSON字符串(在網(wǎng)絡(luò)傳輸中最初的形態(tài)可能是字節(jié)數(shù)組)反射為對應(yīng)類型的對象。
阿里巴巴提供的FastJSON提供了 toJSONString() 和 parseObject() 方法來將 Java 對象與 JSON 相互轉(zhuǎn)換。調(diào)用toJSONString方法即可將對象轉(zhuǎn)換成 JSON 字符串,parseObject 方法則反過來將 JSON 字符串轉(zhuǎn)換成對象。FastJSON的內(nèi)部其實用的就是反射機制。
package com.cmower.common.util;
import java.io.UnsupportedEncodingException;
import org.apache.commons.logging.Log;import org.apache.commons.logging.LogFactory;
import com.alibaba.fastjson.JSON;
@SuppressWarnings("all")public class JsonUtil {
private static Log logger = LogFactory.getLog("json");
public static byte[] objectToByte(Object obj) throws UnsupportedEncodingException {
String jsonStr = JSON.toJSONString(obj);
logger.debug("序列化后數(shù)據(jù):" + jsonStr);
return jsonStr.getBytes("UTF-8");
}
public static T byteToObject(byte[] data, Class obj) throws UnsupportedEncodingException {
String objectString = new String(data, "UTF-8");
logger.debug("反序列化后數(shù)據(jù) : " + objectString);
return JSON.parseObject(objectString, obj);
}
public static Object stringToObject(String data, Class obj) throws UnsupportedEncodingException {
logger.debug("反序列化后數(shù)據(jù) : " + data);
return JSON.parseObject(data, obj);
}}
以上就是java反射的實現(xiàn)和應(yīng)用,詳細(xì)使用情況還得要大家自己使用過才能知道具體要領(lǐng)。如果想閱讀更多相關(guān)內(nèi)容的文章,歡迎關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道!
另外有需要云服務(wù)器可以了解下創(chuàng)新互聯(lián)scvps.cn,海內(nèi)外云服務(wù)器15元起步,三天無理由+7*72小時售后在線,公司持有idc許可證,提供“云服務(wù)器、裸金屬服務(wù)器、高防服務(wù)器、香港服務(wù)器、美國服務(wù)器、虛擬主機、免備案服務(wù)器”等云主機租用服務(wù)以及企業(yè)上云的綜合解決方案,具有“安全穩(wěn)定、簡單易用、服務(wù)可用性高、性價比高”等特點與優(yōu)勢,專為企業(yè)上云打造定制,能夠滿足用戶豐富、多元化的應(yīng)用場景需求。