欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Java獲取調(diào)用當(dāng)前方法的類(lèi)名或方法名(棧堆信息)的四種方式舉例

 更新時(shí)間:2024年09月01日 09:18:46   作者:星月昭銘  
在Java編程中我們經(jīng)常需要在運(yùn)行時(shí)獲取當(dāng)前執(zhí)行的方法名稱(chēng),這在日志記錄、性能監(jiān)控、調(diào)試等方面非常有用,這篇文章主要給大家介紹了關(guān)于Java獲取調(diào)用當(dāng)前方法的類(lèi)名或方法名(棧堆信息)的四種方式,需要的朋友可以參考下

前言

在java代碼中,是可以在運(yùn)行時(shí)通過(guò)某種方式獲取到當(dāng)前方法被誰(shuí)調(diào)用了(調(diào)用鏈路)。目前我所知道的有四種方式(通過(guò)Thread、Throwable、SecurityManager獲?。旅嬷饌€(gè)列出,附有代碼和截圖。

Thread.getAllStackTraces()方式

Thread.getAllStackTraces()可以獲取所有線程的棧堆跟蹤信息,返回值是Map<Thread, StackTraceElement[]>類(lèi)型,可以通過(guò)Thread.currentThread()拿到當(dāng)前Thread對(duì)象,從map集合中獲取棧堆跟蹤信息。StackTraceElement對(duì)象可以拿到調(diào)用者的類(lèi)名、Class對(duì)象、調(diào)用的代碼所在行數(shù)等。

代碼如下:

public class A {
    public static void main(String[] args) {
        B.print();
    }
}

class B {
    public static void print() {
        System.out.println("調(diào)用B#print");
        C.print();
    }
}

class C {
    public static void print() {
        System.out.println("調(diào)用C#print");
        // 獲取所有線程的棧堆跟蹤集合
        Map<Thread, StackTraceElement[]> allStackTraces = Thread.getAllStackTraces();
        // 去map集合里面拿當(dāng)前線程的棧堆跟蹤信息數(shù)組
        StackTraceElement[] stackTraceElements = allStackTraces.get(Thread.currentThread());
        System.out.println(Arrays.toString(stackTraceElements));
    }
}

可以看到,下標(biāo)為4的StackTraceElement對(duì)象就是我們的main方法,之所以出現(xiàn)了下標(biāo)0、1非自定義類(lèi)的棧堆信息是因?yàn)檎{(diào)用了Thread.getAllStackTraces()方法,方法以及方法內(nèi)部也執(zhí)行了其他代碼。

Thread.currentThread().getStackTrace()方式

通過(guò)Thread.currentThread()拿到當(dāng)前的線程,再通過(guò)Thread對(duì)象的getStackTrace()方法獲取線程的棧堆跟蹤信息,getStackTrace()非static方法,所以要先拿到Thread對(duì)象。第一種和第二種方式實(shí)際上都是調(diào)用了private native static StackTraceElement[][] dumpThreads(Thread[] threads)本地方法來(lái)獲取。這兩種方式獲取的話,棧堆跟蹤信息數(shù)組里面都有非自定義類(lèi)的調(diào)用信息(看下標(biāo)0)。

代碼如下:

public class A {
    public static void main(String[] args) {
        B.print();
        System.out.println(Arrays.toString(Thread.currentThread().getStackTrace()));
    }
}

class B {
    public static void print() {
        System.out.println("調(diào)用B#print");
        C.print();
    }
}

class C {
    public static void print() {
        // 獲取當(dāng)前線程,再獲取棧堆跟蹤數(shù)組
        StackTraceElement[] stackTrace = Thread.currentThread().getStackTrace();
        System.out.println(Arrays.toString(stackTrace));
    }
}

new RuntimeException().getStackTrace()方式

實(shí)際上,只要是Throwable的子類(lèi)包括他自身都可以,內(nèi)部也是調(diào)用了一個(gè)本地方法來(lái)獲取的棧堆信息。SpringApplication類(lèi)中的deduceMainApplicationClass()方法就是通過(guò)這種方式來(lái)拿主類(lèi)的Class對(duì)象。拿到了之后就可以通過(guò)for循環(huán)整個(gè)數(shù)組通過(guò)匹配方法名或者類(lèi)型來(lái)拿想要信息了,或者直接通過(guò)整個(gè)數(shù)組拿到整個(gè)調(diào)用鏈路。

代碼如下:

public class A {
    public static void main(String[] args) {
        B.print();
    }
}

class B {
    public static void print() {
        System.out.println("調(diào)用B#print");
        C.print();
    }
}

class C {
    public static void print() {
        StackTraceElement[] stackTrace = new RuntimeException().getStackTrace();
        System.out.println(Arrays.toString(stackTrace));
    }
}

SecurityManager方式(這種可以拿到類(lèi)信息,但是拿不到棧堆信息)

SecurityManager中有一個(gè)getClassContext()方法,這個(gè)方法可以獲取到調(diào)用當(dāng)前方法的Class信息,返回的類(lèi)型是一個(gè)Class數(shù)組,和前面幾種不同的是,這種方式?jīng)]法獲取代碼行數(shù)、調(diào)用的方法等(Class對(duì)象大家應(yīng)該很熟悉了,能拿到哪種信息應(yīng)該都知道)。

由于getClassContext方法是非static的,所以要通過(guò)對(duì)象的方式調(diào)用,但是不能通過(guò)new SecurityManager()的對(duì)象來(lái)獲取,因?yàn)檫@個(gè)方法是一個(gè)protected方法。protected native Class[] getClassContext()。所以需要自定義一個(gè)類(lèi),繼承SecurityManager類(lèi),然后重寫(xiě)getClassContext方法,這樣我們就能在這個(gè)類(lèi)中使用這個(gè)方法了。

這種方式在slf4j中Util#getCallingClass()用到了

代碼如下:

public class A {
    public static void main(String[] args) {
        B.print();
    }
}

class B {
    public static void print() {
        System.out.println("調(diào)用B#print");
        C.print();
    }
}

class C {

    public static void print() {
        // 獲取一個(gè)安全管理器對(duì)象
        ClassContextSecurityManager securityManager = getSecurityManager();
        if (securityManager != null) {
            System.out.println(Arrays.toString(securityManager.getClassContext()));
        }
    }

    static ClassContextSecurityManager getSecurityManager() {
        try {
            return new ClassContextSecurityManager();
        } catch (Throwable throwable) {
            return null;
        }
    }

    /**
     * 自定義的安全管理器,繼承java.lang.SecurityManager類(lèi)
     */
    private static final class ClassContextSecurityManager extends SecurityManager {
    	/**
    	 * 重寫(xiě)getClassContext方法,不然方法在SecurityManager中又被protected修飾,其他類(lèi)還是調(diào)用不了
    	 */
    	@Override
        protected Class<?>[] getClassContext() {
            return super.getClassContext();
        }
    }
}

總結(jié) 

到此這篇關(guān)于Java獲取調(diào)用當(dāng)前方法的類(lèi)名或方法名(棧堆信息)的四種方式的文章就介紹到這了,更多相關(guān)Java獲取調(diào)用當(dāng)前方法類(lèi)名或方法名內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • Logback MDCAdapter日志跟蹤及自定義效果源碼解讀

    Logback MDCAdapter日志跟蹤及自定義效果源碼解讀

    這篇文章主要為大家介紹了Logback MDCAdapter日志跟蹤及自定義效果源碼解讀,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-11-11
  • Java實(shí)現(xiàn)FIFO功能的完整代碼實(shí)踐

    Java實(shí)現(xiàn)FIFO功能的完整代碼實(shí)踐

    在軟件開(kāi)發(fā)中,隊(duì)列(Queue)是一種常見(jiàn)的數(shù)據(jù)結(jié)構(gòu),其特點(diǎn)是先進(jìn)先出(FIFO,First In First Out),FIFO 隊(duì)列在生產(chǎn)者-消費(fèi)者模型、任務(wù)調(diào)度、緩沖區(qū)管理等場(chǎng)景中具有廣泛的應(yīng)用,本文給大家介紹了Java實(shí)現(xiàn)FIFO功能的完整代碼實(shí)踐,需要的朋友可以參考下
    2025-03-03
  • SpringCloud中Gateway的使用教程詳解

    SpringCloud中Gateway的使用教程詳解

    SpringCloud?Gateway是Spring體系內(nèi)的一個(gè)全新項(xiàng)目,它旨在為微服務(wù)架構(gòu)提供一種簡(jiǎn)單有效的統(tǒng)一的API路由管理方式。本文就來(lái)為大家詳細(xì)講講Gateway的使用教程,需要的可以參考一下
    2022-08-08
  • spring security環(huán)境搭建

    spring security環(huán)境搭建

    本文通過(guò)代碼給大家介紹了spring security環(huán)境搭建的詳細(xì)教程,非常不錯(cuò),具有參考借鑒價(jià)值,需要的朋友參考下吧
    2017-09-09
  • spring retry 配置及使用教程

    spring retry 配置及使用教程

    這篇文章主要介紹了spring retry 配置及使用教程,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2024-01-01
  • java自定義jar包讓jmeter使用的方法

    java自定義jar包讓jmeter使用的方法

    在本篇文章里小編給大家整理了一篇關(guān)于java自定義jar包讓jmeter使用的方法以及實(shí)例代碼,需要的朋友們參考下。
    2019-10-10
  • Java 實(shí)戰(zhàn)項(xiàng)目錘煉之在線美食網(wǎng)站系統(tǒng)的實(shí)現(xiàn)流程

    Java 實(shí)戰(zhàn)項(xiàng)目錘煉之在線美食網(wǎng)站系統(tǒng)的實(shí)現(xiàn)流程

    讀萬(wàn)卷書(shū)不如行萬(wàn)里路,只學(xué)書(shū)上的理論是遠(yuǎn)遠(yuǎn)不夠的,只有在實(shí)戰(zhàn)中才能獲得能力的提升,本篇文章手把手帶你用java+SSM+jsp+mysql+maven實(shí)現(xiàn)一個(gè)在線美食網(wǎng)站系統(tǒng),大家可以在過(guò)程中查缺補(bǔ)漏,提升水平
    2021-11-11
  • 深入理解Java的接口與抽象類(lèi)

    深入理解Java的接口與抽象類(lèi)

    本文主要介紹java 的接口和抽象類(lèi),對(duì)接口和抽象類(lèi)進(jìn)行介紹對(duì)比,深入理解,有需要的小伙伴可以參考下
    2016-07-07
  • 淺談Spring事務(wù)傳播行為實(shí)戰(zhàn)

    淺談Spring事務(wù)傳播行為實(shí)戰(zhàn)

    這篇文章主要介紹了淺談Spring事務(wù)傳播行為實(shí)戰(zhàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2019-09-09
  • SpringBoot 二維碼生成base64并上傳OSS的實(shí)現(xiàn)示例

    SpringBoot 二維碼生成base64并上傳OSS的實(shí)現(xiàn)示例

    本文主要介紹了SpringBoot 二維碼生成base64并上傳OSS的實(shí)現(xiàn)示例,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2022-05-05

最新評(píng)論