Java反射

来源:https://www.cnblogs.com/lewis09/archive/2019/01/18/10288854.html
-Advertisement-
Play Games

Java反射機制在程式運行時,對於任意一個類,都能夠知道這個類的所有屬性和方法;對於任意一個對象,都能夠調用它的任意一個方法和屬性。這種動態的獲取信息以及動態調用對象的方法 的功能稱為java的反射機制。 首先你需要瞭解類載入的過程,這裡我們簡單提一下(載入-驗證-準備-解析-初始化),反射是靠JV ...


Java反射機制在程式運行時,對於任意一個類,都能夠知道這個類的所有屬性和方法;對於任意一個對象,都能夠調用它的任意一個方法和屬性。這種動態的獲取信息以及動態調用對象的方法 的功能稱為java的反射機制。

首先你需要瞭解類載入的過程,這裡我們簡單提一下(載入-驗證-準備-解析-初始化),反射是靠JVM和Class相關類實現的。

按照這個例子,我們調試下看看具體實現。

@Data
public class Person {
    private String name;

    public static void main(String[] args) throws Exception {
        Person person = new Person();
        person.setName("lewis");

        for (int i = 0; i < 16; i++) {
            Method method = Person.class.getMethod("getName");
            System.out.println(method.invoke(person));
        }
    }
}

 

@CallerSensitive
public Object invoke(Object obj, Object... args)
    throws IllegalAccessException, IllegalArgumentException,
       InvocationTargetException
{
    if (!override) {
        // 檢查方法是否為public
        if (!Reflection.quickCheckMemberAccess(clazz, modifiers)) {
            Class<?> caller = Reflection.getCallerClass();
            // 許可權校驗
            checkAccess(caller, clazz, obj, modifiers);
        }
    }
    // MethodAccessor實現有兩個版本,一個是Java實現的,另一個是JNI實現的
    MethodAccessor ma = methodAccessor;             // read volatile
    if (ma == null) {
        ma = acquireMethodAccessor();
    }
    return ma.invoke(obj, args);
}

我們上面提到了 MethodAccessor 有兩個實現,Java版本和JNI版本(就是java native),

Java實現的版本在初始化時需要較多時間,但長久來說性能較好;JNI版本正好相反,啟動時相對較快,但運行時間長了之後速度就比不過Java版了。   為了儘可能地減少性能損耗,HotSpot JDK採用“inflation”的技巧:讓Java方法在被反射調用時,開頭若幹次使用JNI版,等反射調用次數超過閾值(15)時則生成一個專用的MethodAccessor實現類,生成其中的invoke()方法的位元組碼,以後對該Java方法的反射調用就會使用Java版本。    ReflectionFactory.newMethodAccessor()生產MethodAccessor對象的邏輯,一開始(JNI版)會生產NativeMethodAccessorImpl和DelegatingMethodAccessorImpl兩個對象。
    public MethodAccessor newMethodAccessor(Method var1) {
        checkInitted();
        if (noInflation && !ReflectUtil.isVMAnonymousClass(var1.getDeclaringClass())) {
            return (new MethodAccessorGenerator()).generateMethod(var1.getDeclaringClass(), var1.getName(), var1.getParameterTypes(), var1.getReturnType(), var1.getExceptionTypes(), var1.getModifiers());
        } else {
            NativeMethodAccessorImpl var2 = new NativeMethodAccessorImpl(var1);
            DelegatingMethodAccessorImpl var3 = new DelegatingMethodAccessorImpl(var2);
            var2.setParent(var3);
            return var3;
        }
    }
DelegatingMethodAccessorImpl的源碼如下: 這是一個中間層,方便在JNI版本與Java版的MethodAccessor之間實現切換。
class DelegatingMethodAccessorImpl extends MethodAccessorImpl {
    private MethodAccessorImpl delegate;

    DelegatingMethodAccessorImpl(MethodAccessorImpl var1) {
        this.setDelegate(var1);
    }

    public Object invoke(Object var1, Object[] var2) throws IllegalArgumentException, InvocationTargetException {
        return this.delegate.invoke(var1, var2);
    }

    void setDelegate(MethodAccessorImpl var1) {
        this.delegate = var1;
    }
}

來看看NativeMethodAccessorImpl實現,超過15次以後調用反射,就會通過我們上面提到的中間層 DelegatingMethodAccessorImpl 所引用的 MethodAccessor 都是java 版。

class NativeMethodAccessorImpl extends MethodAccessorImpl {
    private final Method method;
    private DelegatingMethodAccessorImpl parent;
    private int numInvocations;

    NativeMethodAccessorImpl(Method var1) {
        this.method = var1;
    }

    public Object invoke(Object var1, Object[] var2) throws IllegalArgumentException, InvocationTargetException {
        // 方法被調用時,程式調用計數器都會增加1,看看是否超過閾值
        if (++this.numInvocations > ReflectionFactory.inflationThreshold() && !ReflectUtil.isVMAnonymousClass(this.method.getDeclaringClass())) {
            // 超過15次 則調用MethodAccessorGenerator.generateMethod()來生成Java版的MethodAccessor的實現類
            // 並且改變通過中間層,後續DelegatingMethodAccessorImpl所引用的MethodAccessor改為Java版
            MethodAccessorImpl var3 = (MethodAccessorImpl)(new MethodAccessorGenerator()).generateMethod(this.method.getDeclaringClass(), this.method.getName(), this.method.getParameterTypes(), this.method.getReturnType(), this.method.getExceptionTypes(), this.method.getModifiers());
            this.parent.setDelegate(var3);
        }
        // native版本,JNI方式調用
        return invoke0(this.method, var1, var2);
    }

    void setParent(DelegatingMethodAccessorImpl var1) {
        this.parent = var1;
    }

    private static native Object invoke0(Method var0, Object var1, Object[] var2);
}

在預設情況下,方法的反射調用為委派實現,委派給本地實現來進行方法調用。在調用超過 15 次之後,委派實現便會將委派對象切換至動態實現。這個動態的位元組碼是在Java運行過程中通過ASM自動生成的,它將直接使用 invoke 指令來調用目標方法。

繼續查看代碼,可以看到sun.reflect.MethodAccessorGenerator#generate的實現是調用asm位元組碼增強工具來生成類,此過程較長,不在此列出。在該方法的最後,我們發現有這樣一個操作sun.reflect.ClassDefiner#defineClass,查看其源碼

static Class<?> defineClass(String name, byte[] bytes, int off, int len,
                                final ClassLoader parentClassLoader)
    {
        // 創建一個DelegatingClassLoader用來載入生成的類
        ClassLoader newLoader = AccessController.doPrivileged(
            new PrivilegedAction<ClassLoader>() {
                public ClassLoader run() {
                        return new DelegatingClassLoader(parentClassLoader);
                    }
                });
        return unsafe.defineClass(name, bytes, off, len, newLoader, null);
}

 

參考:

一個關於log4j2的高併發問題

 

 


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

-Advertisement-
Play Games
更多相關文章
  • import pandas as pd import numpy as np import names ''' 寫在前面的話: 1、series與array類型的不同之處為series有索引,而另一個沒有;series中的數據必須是一維的,而array類型不一定 2、可以把series看成一個定長的... ...
  • 1.自定義實現replace方法 Python replace() 方法把字元串中的 old(舊字元串) 替換成 neange(新字元串),如果指定第三個參數max,則替換不超過 max 次。考慮old與nein的長度不一樣的情況,如old = 'is';new = 'was' 思路: 1.先找出字 ...
  • 前言 寫了兩年多的博客了,我想整理我的博客,問題是,得一個個打開,之後複製粘貼,嫌這樣太麻煩,於是便找到了博客園的備份功能。 但是這個備份功能下載下來的只是一個xml文件,我想把每一篇博文都轉為一個md文件,於是便有了這個Java小工具。 工具下載 "BlogBackupTool" 使用說明 從博客 ...
  • Python基礎篇 1:為什麼學習Python 2:通過什麼途徑學習Python 3:談談對Python和其他語言的區別 Python的優勢: 4:簡述解釋型和編譯型編程語言 5:Python的解釋器種類以及相關特點? 6:位和位元組的關係 7:b、B、KB、MB、GB的關係 8:PE8規範 9:通過 ...
  • 根據上面獲取的數據開始創建java文件 終於開始要創建java文件了。 但是~在創建java文件的時候要先吧之前獲取的數稍微處理一下,將sql中的格式轉換為java中的格式。比如屬性名稱,數據類型,class名稱之類的,現在開始~ 將表名稱轉換為合適的class名稱 就是首字母大寫,駝峰式的命名規範 ...
  • 1.電腦一直在演化,64核,128核等等,但是我們依舊在使用為單核設計的技術編程2.Go語言讓分享自己的代碼包更容易3.Go語言重新思考傳統的面向對象,提供了更高效的復用代碼手段4.Go不僅提供高性能而且開發更快速5.Go語法簡潔,編譯速度快,內置併發,自帶垃圾回收器6.Go編譯器只關註直接被引用 ...
  • 1.利用Python實現Pig Latin字母游戲 “Pig Latin”是一個英語兒童文字改寫游戲,整個游戲遵從下述規則:a. 母音字母是‘a’、‘e’、‘i’、‘o’、‘u’。字母‘y’在不是第一個字母的情況下,也被視作母音字母。其他字母均為輔音字母。例如,單詞“yearly”有三個母音字母(分 ...
  • 1、變數: 2、字元串:用 “ ” 或 ' ' 標註的。 3、列表:[ ] 4、del 和 pop 的區別: 判斷何時使用:當從列表中刪除元素後不再使用,則del,若後續還使用則pop() 例: #del #pop() ['lele', 'ningning', 'tuotuo', 'yangyang ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...