1.java 9以前堆棧遍歷
到目前為止,官方解決方案是獲取當(dāng)前線程并調(diào)用其getstacktrace()
方法:
1
|
stacktraceelement[] stacktraceelements = thread.currentthread().getstacktrace(); |
另一個(gè)智能解決方案涉及.拋出異常并從中提取堆棧跟蹤信息。 但是,無(wú)法操縱結(jié)果,它會(huì)立即打印出來(lái):
1
|
new exception().printstacktrace(); |
兩種解決方案都存在同樣的問(wèn)題——它們都急切地捕獲整個(gè)堆棧的快照,可不方便使用。
2. jep-259: stack-walking api
jep-259應(yīng)該解決這些問(wèn)題,而且確實(shí)如此。 新的api提供了一種使用stream api懶惰地遍歷堆棧跟蹤的便捷方法。
我們可以像這樣輕松地創(chuàng)建stackwalker 實(shí)例:
1
|
stackwalker stack = stackwalker.getinstance(); |
此外,我們可以提供一些初始選項(xiàng):
1
|
stackwalker = stackwalker.getinstance(stackwalker.option.retain_class_reference); |
如果我們想要遍歷整個(gè)堆棧,那只需要調(diào)用foreach()
方法:
stack.foreach(system.out::println);
3. stackwalker.stackframe
如果我們查看java 1.4的stacktraceelement——它幾乎是一個(gè)包含有關(guān)聲明類、方法名、類加載器名等的詳細(xì)字符串信息。
stackwalker.stackframe是一個(gè)更加類型安全友好的升級(jí),在其上面提到了豐富的方法:
public class<?> getdeclaringclass();
public methodtype getmethodtype();
…甚至可這樣:
public stacktraceelement tostacktraceelement();
4.示例
讓我們將前面那些付諸實(shí)踐,來(lái)創(chuàng)建一個(gè)簡(jiǎn)單的調(diào)用層次結(jié)構(gòu)
1
2
3
4
5
6
7
8
9
10
11
12
|
(代碼包和類名: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); } |
如果我們?cè)趇de中(jshell運(yùn)行顯示會(huì)不一樣,這個(gè)它的處理模式有關(guān))中運(yù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.高級(jí)特性
如果我們想利用惰性或幀過(guò)濾,我們可以使用另一個(gè)名為walk()的專用api方法,它允許我們使用stream api來(lái)方便地遍歷堆棧。 在閱讀本文時(shí),您可能想象walk()方法只是返回一個(gè)stream實(shí)例。事實(shí)并非如此。
實(shí)際的簽名是:
public <t> t walk(function<? super stream<stackframe>, ? extends t> function)
還有一個(gè)很好的理由使它成為這種方式——堆棧需要被凍結(jié)以便遍歷它,并且這發(fā)生在walk()方法調(diào)用的范圍內(nèi) - 所以使用基于函數(shù)接口的模板方法實(shí)現(xiàn)這一目標(biāo)是有意義的 。
即使你試圖通過(guò)返回一個(gè)stream實(shí)例來(lái)欺騙它,它也無(wú)法使用(自己試試看!)。
一旦我們知道了這個(gè)限制,我們只受我們的想象力和stream api功能的約束。例如,我們可以優(yōu)雅地跳過(guò)一些幀,然后挑選第一個(gè)遇到的幀:
1
2
3
4
|
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.完整代碼清單
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
|
/* *測(cè)試堆棧遍歷 */ 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); //第一次運(yùn)行,注釋掉 //bar();//第二次運(yùn)行注釋掉 } private static void bar() { java.lang.stackwalker .getinstance(java.lang.stackwalker.option.retain_class_reference) .foreach(system.out::println); } } |
總結(jié)
以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,謝謝大家對(duì)服務(wù)器之家的支持。如果你想了解更多相關(guān)內(nèi)容請(qǐng)查看下面相關(guān)鏈接
原文鏈接:https://www.toutiao.com/a6600896038721028622/