【JavaSE】反射的總結

来源:https://www.cnblogs.com/popo33/archive/2020/05/25/12961789.html
-Advertisement-
Play Games

@ 一、類的載入概述 載入 就是指將class文件讀入記憶體,併為之創建一個Class對象 任何類被使用時系統都會建立一個Class對象 連接 驗證:是否有正確的內部結構,並和其他類協調一致 準備:負責為類的靜態成員分配記憶體,並設置預設初始化值 解析:將類的二進位數據中的符號引用替換為直接引用 初始化 ...


@目錄

一、類的載入概述

載入

  • 就是指將class文件讀入記憶體,併為之創建一個Class對象
  • 任何類被使用時系統都會建立一個Class對象

連接

  • 驗證:是否有正確的內部結構,並和其他類協調一致
  • 準備:負責為類的靜態成員分配記憶體,並設置預設初始化值
  • 解析:將類的二進位數據中的符號引用替換為直接引用

初始化

  • 就是我們之前講過的初始化步驟

類的初始化時機(什麼時候被載入)

  • 創建類的實例
  • 用到類的靜態變數
  • 用到類的靜態方法
  • 使用反射方式強制創建某個類或介面對應的java.lang.Class對象
  • 初始化某個類的子類 直接使用java.exe命令來運行某個主類

類載入器

引導類載入器(Bootstrap ClassLoader)

  • 引導類載入器是jvm在運行時,內嵌在jvm中的一段特殊的用來載入java核心類庫的C++代碼。String.class 對象就是由引導類載入器載入的,引導類載入器具體載入哪些核心代碼可以通過獲取值為 "sun.boot.class.path" 的系統屬性獲得。引導類載入器不是java原生代碼編寫的,所以其也不是java.lang.ClassLoader類的實例,其沒有getParent方法。

拓展類載入器(Extension ClassLoader)

  • 拓展類載入器用來載入jvm實現的一個拓展目錄,該目錄下的所有java類都由此類載入器載入。此路徑可以通過獲取"java.ext.dirs"的系統屬性獲得。拓展類載入器就是java.lang.ClassLoader類的一個實例,其getParent方法返回的是引導類載入器(在 HotSpot虛擬機中用null表示引導類載入)。

應用類載入器(Application ClassLoader)

  • 應用類載入器又稱為系統類載入器,開發者可用通過 java.lang.ClassLoader.getSystemClassLoader()方法獲得此類載入器的實例,系統類載入器也因此得名。其主要負責載入程式開發者自己編寫的java類。一般來說,java應用都是用此類載入器完成載入的,可以通過獲取"java.class.path"的系統屬性(也就是我們常說的classpath)來獲取應用類載入器載入的類路徑。應用類載入器是java.lang.ClassLoader類的一個實例,其getParent方法返回的是拓展類載入器。

二、反射的基本使用

2.1 獲取Class位元組碼對象

getClass()
.class
Class.forName("文件全路徑")

2.2 獲取構造方法

獲取所有

public Constructor[] getConstructors()
public Constructor[] getDeclaredConstructors()

獲取單個

public Constructor<T> getConstructor(Class<?>...parameterTypes);//帶有參數
getDeclaredConstructor();//獲取私有構造方法對象
con.setAccessible(true;//指反射的對象在使用時應取消java語言訪問檢查,暴力訪問
con.newInstance();//使用構造方法對象表示的構造方法來創建該構造方法的聲明類的新實例

代碼:獲取無參構造函數

public class ReflexDemo01 {

    public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, IllegalAccessException, InvocationTargetException, InstantiationException {
        // 獲取位元組碼對象
        Class person = Class.forName("Person");
        // 獲取構造方法
        Constructor constructor = person.getConstructor();
        // 實例化對象
        Object obj = constructor.newInstance();

        System.out.println(obj);//Person@1b6d3586
    }
}

代碼:獲取帶參構造函數

 public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, IllegalAccessException, InvocationTargetException, InstantiationException {
        Class person = Class.forName("Person");
        Constructor constructor = person.getConstructor(String.class, String.class, int.class);
        Object obj = constructor.newInstance("張三","男",20);
        System.out.println(obj);
    }

代碼:獲取私有構造函數

 public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, IllegalAccessException, InvocationTargetException, InstantiationException {
        Class person = Class.forName("Person");
        
        Constructor constructor = person.getDeclaredConstructor(String.class, String.class, int.class);
      
        constructor.setAccessible(true);
        
        Object obj = constructor.newInstance("張三","男",20);
       
        System.out.println(obj);
    }

註意:如果不設置 constructor.setAccessible(true);
會報錯:java.lang.IllegalAccessException: Class ReflexDemo01 can not access a
member of class Person with modifiers "private"

2.3 獲取成員變數

獲取所有

  • Field[] getFields()
  • Field[] getDeclaredFields()

獲取單個

  • Filed getField("成員變數名")
  • Field getDeclaredField()
  • field.setAccessible(true)

2.4 獲取成員方法並使用

  • getMethods
  • getDeclaredMethods
  • getMethod
  • getDeclaredMethod
  • method.setAccessible(true)
public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, IllegalAccessException, InvocationTargetException, InstantiationException {
        Class person = Class.forName("Person");

        Constructor constructor = person.getDeclaredConstructor(String.class, String.class, int.class);

        constructor.setAccessible(true);

        Object obj = constructor.newInstance("張三","男",20);

        // Method[] methods = c.getMethods(); // 獲取自己的包括父親的公共方法
        // Method[] methods = c.getDeclaredMethods(); // 獲取自己的所有的方法
        //Method[] methods = c.getMethod(String name,Class<?>... parameterTypes)
        // 第一個參數表示的方法名,第二個參數表示的是方法的參數的class類型
        Method method01 = person.getMethod("method01");
        Method method02 = person.getMethod("method02", String.class);

        Method method03 = person.getDeclaredMethod("method03", String.class, int.class);
        method03.setAccessible(true);

        method01.invoke(obj);
        method02.invoke(obj,"hello");
        Object resultStr = method03.invoke(obj, "hello", 123);
        System.out.println(resultStr);

    }

三、案例

3.1 通過反射運行配置文件

/**
 * 通過反射運行配置文件
 */
public class ReflexDemo02 {
    public static void main(String[] args) throws IOException, ClassNotFoundException, NoSuchMethodException, IllegalAccessException, InvocationTargetException, InstantiationException {

        // 讀取鍵值對數據
        Properties prop = new Properties();
        FileReader fr = new FileReader("F:\\Projects\\idea_projects\\javaseTest\\09reflex\\src\\class.txt");
        prop.load(fr);
        fr.close();
        // 獲取文件數據
        String className = prop.getProperty("className");
        String methodName = prop.getProperty("methodName");

        // 反射
        Class c = Class.forName(className);

        Constructor constructor = c.getConstructor();

        Object obj = constructor.newInstance();

        Method method = c.getMethod(methodName);

        method.invoke(obj);


    }
}

3.2 通過反射越過泛型檢查

/**
 * 通過反射越過泛型檢查
 */
public class ReflexDemo03 {
    public static void main(String[] args) throws NoSuchMethodException, InvocationTargetException, IllegalAccessException {

        // 創建集合
        ArrayList<Integer> arrayList = new ArrayList<Integer>();

        // arrayList.add(123);
       //  arrayList.add("hello");//直接調用編譯期就會報錯

        Class c = arrayList.getClass();
        Method m = c.getMethod("add", Object.class);

        m.invoke(arrayList,"hello");
        m.invoke(arrayList,"world");
        m.invoke(arrayList,"java");

        System.out.println(arrayList);//[hello, world, java]


    }
}

3.3 通過反射寫一個Tool工具類,設置任意對象的任意屬性

public class PropertyTools {
    /**
     * 給傳入對象設置屬性
     */
    public void setProperty(Object obj,String propertyName,Object value) throws NoSuchFieldException, IllegalAccessException {

        // 根據對象獲取位元組碼對象
        Class c = obj.getClass();
        // 根據名稱獲取成員變數
        Field field = c.getDeclaredField(propertyName);
        // 取消訪問檢查
        field.setAccessible(true);
        // 給對象的成員變數賦值為指定的值
        field.set(obj,value);

    }

    /**
     * 測試該工具類
     */
    public static void main(String[] args) throws NoSuchFieldException, IllegalAccessException {
        Person person = new Person();
        PropertyTools tools = new PropertyTools();
        tools.setProperty(person,"name","林青霞");
        tools.setProperty(person,"sex","女");
        tools.setProperty(person,"age",20);

        System.out.println(person);//Person{name='林青霞', sex='女', age=20}
    }


}

感覺這個案例很能說明:反射破壞了封裝性


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 一、JML初探 ​ 作為一種形式化語言,可以約束 代碼中類和方法的狀態和行為形成規格,通過將一系列具體代碼實現抽象成明確的行為介面,可以形成一種契約式編程模式, 設計者無需考慮實際的數據結構與演算法,可以聚焦於程式的整體邏輯, 形式化語言的無二義性能讓實現者準確理解介面功能,根據問題需要選擇合適的實現 ...
  • 一、基本概念 門面模式(外觀模式)是對象的結構模式,外部與一個子系統的通信必須通過一個統一的門面對象進行。門面模式提供一個高層次的介面,使得子系統更易於使用。 二、通俗解釋 FACADE門面模式:我有一個專業的Nikon相機,我就喜歡自己手動調光圈、快門,這樣照出來的照片才專業,但MM可不懂這些,教 ...
  • 一、基本概念 裝飾模式又名包裝(Wrapper)模式。裝飾模式以對客戶端透明的方式擴展對象的功能,是繼承關係的一個替代方案。 二、通俗解釋 DECORATOR裝飾模式:Mary過完輪到Sarly過生日,還是不要叫她自己挑了,不然這個月伙食費肯定玩完,拿出我去年在華山頂上照的照片,在背面寫上“最好的的 ...
  • 一、基本概念 合成模式屬於對象的結構模式,有時又叫做“部分——整體”模式。合成模式將對象組織到樹結構中,可以用來描述整體與部分的關係。合成模式可以使客戶端將單純元素與複合元素同等看待。 二、通俗解釋 COMPOSITE合成模式:Mary今天過生日。“我過生日,你要送我一件禮物。”“嗯,好吧,去商店, ...
  • 一、基本概念 橋梁模式(Bridge)是對象的結構模式。又稱為柄體(Handle and Body)模式或介面(Interface)模式。橋梁模式的用意是“將抽象化(Abstraction)與實現化(Implementation)脫耦,使得二者可以獨立地變化”。 這句話有三個關鍵詞,也就是抽象化、實 ...
  • 一、基本概念 適配器模式是將某個類的介面轉換成客戶端期望的另一個介面表示,目的是消除由於介面不匹配所造成的的類的相容性問題。 二、通俗解釋 ADAPTER 適配器模式:在朋友聚會上碰到了一個美女Sarah,從香港來的,可我不會說粵語,她不會說普通話,只好求助於我的朋友kent了,他作為我和Sarah ...
  • 我們在使用Spring框架中,特別是框架級的功能,經常看到有@Import導入功能, ​ 我就介紹下它能導入什麼,首先聲明下@Import是註解,導入類型可分為三類: 1. 導入配置 @Configuration,類似於spring早期版本2.5的import xml文件一樣, <?xml vers ...
  • 一、不同環境下的C 1.一個源程式inform.c編譯生成a.out,然後可以直接運行這個可執行程式,如果要存儲這個可執行文件,應該把它重命名,否則,就會被下一次編譯程式時生成的新a.out所替換。 2.如何處理目標代碼,C編譯器會創建一個inform.o,然而卻找不到這個文件,因為一旦鏈接器生成了 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...