1.Java 9以前堆棧遍歷
到目前為止,官方解決方案是獲取當(dāng)前線程并調(diào)用其getStackTrace()
方法:
專注于為中小企業(yè)提供成都網(wǎng)站建設(shè)、網(wǎng)站建設(shè)服務(wù),電腦端+手機端+微信端的三站合一,更高效的管理,為中小企業(yè)衡水免費做網(wǎng)站提供優(yōu)質(zhì)的服務(wù)。我們立足成都,凝聚了一批互聯(lián)網(wǎng)行業(yè)人才,有力地推動了上1000家企業(yè)的穩(wěn)健成長,幫助中小企業(yè)通過網(wǎng)站建設(shè)實現(xiàn)規(guī)模擴充和轉(zhuǎn)變。
StackTraceElement[] stackTraceElements = Thread.currentThread().getStackTrace();
另一個智能解決方案涉及.拋出異常并從中提取堆棧跟蹤信息。 但是,無法操縱結(jié)果,它會立即打印出來:
new Exception().printStackTrace();
兩種解決方案都存在同樣的問題——它們都急切地捕獲整個堆棧的快照,可不方便使用。
2. JEP-259: Stack-Walking API
JEP-259應(yīng)該解決這些問題,而且確實如此。 新的API提供了一種使用Stream API懶惰地遍歷堆棧跟蹤的便捷方法。
我們可以像這樣輕松地創(chuàng)建StackWalker實例:
StackWalker stack = StackWalker.getInstance();
此外,我們可以提供一些初始選項:
StackWalker = StackWalker.getInstance(StackWalker.Option.RETAIN_CLASS_REFERENCE);
如果我們想要遍歷整個堆棧,那只需要調(diào)用forEach()
方法:
stack.forEach(System.out::println);
3. StackWalker.StackFrame
如果我們查看Java 1.4的StackTraceElement——它幾乎是一個包含有關(guān)聲明類、方法名、類加載器名等的詳細字符串信息。
StackWalker.StackFrame是一個更加類型安全友好的升級,在其上面提到了豐富的方法:
public Class<?> getDeclaringClass();
public MethodType getMethodType();
…甚至可這樣:
public StackTraceElement toStackTraceElement();
4.示例
讓我們將前面那些付諸實踐,來創(chuàng)建一個簡單的調(diào)用層次結(jié)構(gòu)
(代碼包和類名:com.nd.stackwalker. StackWalker): public static void main(String[] args) { foo(); } private static void foo() { bar(); } private static void bar() { java.lang.StackWalker .getInstance(java.lang.StackWalker.Option.RETAIN_CLASS_REFERENCE) .forEach(System.out::println); }
如果我們在IDE中(jshell運行顯示會不一樣,這個它的處理模式有關(guān))中運行它,結(jié)果將是(注意堆棧元素的順序):
com.nd.stackwalker.StackWalker.bar(StackWalker.java:22)
com.nd.stackwalker.StackWalker.foo(StackWalker.java:17)
com.nd.stackwalker.StackWalker.main(StackWalker.java:14)
5.高級特性
如果我們想利用惰性或幀過濾,我們可以使用另一個名為walk()的專用API方法,它允許我們使用Stream API來方便地遍歷堆棧。 在閱讀本文時,您可能想象walk()方法只是返回一個Stream實例。事實并非如此。
實際的簽名是:
public
還有一個很好的理由使它成為這種方式——堆棧需要被凍結(jié)以便遍歷它,并且這發(fā)生在walk()方法調(diào)用的范圍內(nèi) - 所以使用基于函數(shù)接口的模板方法實現(xiàn)這一目標是有意義的 。
即使你試圖通過返回一個Stream實例來欺騙它,它也無法使用(自己試試看?。?br />一旦我們知道了這個限制,我們只受我們的想象力和Stream API功能的約束。例如,我們可以優(yōu)雅地跳過一些幀,然后挑選第一個遇到的幀:
java.lang.StackWalker .getInstance(java.lang.StackWalker.Option.RETAIN_CLASS_REFERENCE) .walk(s -> s.skip(1).limit(1).collect(Collectors.toList())) .forEach(System.out::println);
// 結(jié)果如下:
com.nd.stackwalker.StackWalker.main(StackWalker.java:17)
6.完整代碼清單
/* *測試堆棧遍歷 */ package com.nd.stackwalker; import java.util.stream.Collectors; /** * * @author Solo Cui */ public class StackWalker { public static void main(String[] args) { foo(); } private static void foo() { java.lang.StackWalker .getInstance(java.lang.StackWalker.Option.RETAIN_CLASS_REFERENCE) .walk(s -> s.skip(1).limit(1).collect(Collectors.toList())) .forEach(System.out::println);//第一次運行,注釋掉 //bar();//第二次運行注釋掉 } private static void bar() { java.lang.StackWalker .getInstance(java.lang.StackWalker.Option.RETAIN_CLASS_REFERENCE) .forEach(System.out::println); } }
總結(jié)
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,謝謝大家對創(chuàng)新互聯(lián)的支持。如果你想了解更多相關(guān)內(nèi)容請查看下面相關(guān)鏈接