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

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

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

前言

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

Thread.getAllStackTraces()方式

Thread.getAllStackTraces()可以獲取所有線程的棧堆跟蹤信息,返回值是Map<Thread, StackTraceElement[]>類型,可以通過Thread.currentThread()拿到當(dāng)前Thread對象,從map集合中獲取棧堆跟蹤信息。StackTraceElement對象可以拿到調(diào)用者的類名、Class對象、調(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對象就是我們的main方法,之所以出現(xiàn)了下標(biāo)0、1非自定義類的棧堆信息是因為調(diào)用了Thread.getAllStackTraces()方法,方法以及方法內(nèi)部也執(zhí)行了其他代碼。

Thread.currentThread().getStackTrace()方式

通過Thread.currentThread()拿到當(dāng)前的線程,再通過Thread對象的getStackTrace()方法獲取線程的棧堆跟蹤信息,getStackTrace()非static方法,所以要先拿到Thread對象。第一種和第二種方式實際上都是調(diào)用了private native static StackTraceElement[][] dumpThreads(Thread[] threads)本地方法來獲取。這兩種方式獲取的話,棧堆跟蹤信息數(shù)組里面都有非自定義類的調(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()方式

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

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

由于getClassContext方法是非static的,所以要通過對象的方式調(diào)用,但是不能通過new SecurityManager()的對象來獲取,因為這個方法是一個protected方法。protected native Class[] getClassContext()。所以需要自定義一個類,繼承SecurityManager類,然后重寫getClassContext方法,這樣我們就能在這個類中使用這個方法了。

這種方式在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() {
        // 獲取一個安全管理器對象
        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類
     */
    private static final class ClassContextSecurityManager extends SecurityManager {
    	/**
    	 * 重寫getClassContext方法,不然方法在SecurityManager中又被protected修飾,其他類還是調(diào)用不了
    	 */
    	@Override
        protected Class<?>[] getClassContext() {
            return super.getClassContext();
        }
    }
}

總結(jié) 

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

相關(guān)文章

  • java實現(xiàn)voctor按指定方式排序示例分享

    java實現(xiàn)voctor按指定方式排序示例分享

    這篇文章主要介紹了java實現(xiàn)voctor按指定方式排序示例,需要的朋友可以參考下
    2014-03-03
  • Java中的泛型

    Java中的泛型

    這篇文章主要介紹為何要泛型,如何使用泛型,自定義泛型的方法,泛型類的子類等多方面介紹了JAVA中的泛型,需要的小伙伴請看下文
    2021-08-08
  • Java17中record替代Lombok部分功能使用場景探究

    Java17中record替代Lombok部分功能使用場景探究

    這篇文章主要介紹了使用Java17中的record替代Lombok的部分功能,本文來為大家小小的總結(jié)下,我們可以在哪些地方,利用record來替換Lombok
    2024-01-01
  • Redis Lettuce連接redis集群實現(xiàn)過程詳細(xì)講解

    Redis Lettuce連接redis集群實現(xiàn)過程詳細(xì)講解

    這篇文章主要介紹了Redis Lettuce連接redis集群實現(xiàn)過程,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)吧
    2023-01-01
  • java解析xml之jdom解析xml示例分享

    java解析xml之jdom解析xml示例分享

    JDOM是專門為Java打造的API,JDOM采用了Java中的Collection架構(gòu)來封裝集合,是Java愛好者更加熟悉的模式,下面看使用示例
    2014-01-01
  • Spring Batch遠(yuǎn)程分區(qū)的本地Jar包模式的代碼詳解

    Spring Batch遠(yuǎn)程分區(qū)的本地Jar包模式的代碼詳解

    這篇文章主要介紹了Spring Batch遠(yuǎn)程分區(qū)的本地Jar包模式,本文通過實例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2020-09-09
  • Java多線程的sleep休眠的實現(xiàn)

    Java多線程的sleep休眠的實現(xiàn)

    本文主要介紹了Java多線程的sleep休眠的實現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2023-06-06
  • springBoot系列常用注解(小結(jié))

    springBoot系列常用注解(小結(jié))

    這篇文章主要介紹了springBoot系列常用注解,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2021-04-04
  • idea同步y(tǒng)api插件的詳細(xì)步驟

    idea同步y(tǒng)api插件的詳細(xì)步驟

    yapi是一個很好的接口文檔維護(hù)工具,其swagger功能,可將接口信息同步到y(tǒng)api平臺上,這篇文章主要介紹了idea同步y(tǒng)api插件,需要的朋友可以參考下
    2024-04-04
  • IDEA創(chuàng)建SpringBoot父子Module項目的實現(xiàn)

    IDEA創(chuàng)建SpringBoot父子Module項目的實現(xiàn)

    本文主要介紹了IDEA創(chuàng)建SpringBoot父子Module項目的實現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2023-05-05

最新評論