Java中使用StackWalker和Stream API进行堆栈遍历
作者:牛旦教育IT课堂 发布时间:2023-04-12 11:29:07
1.Java 9以前堆栈遍历
到目前为止,官方解决方案是获取当前线程并调用其getStackTrace()
方法:
StackTraceElement[] stackTraceElements = Thread.currentThread().getStackTrace();
另一个智能解决方案涉及.抛出异常并从中提取堆栈跟踪信息。 但是,无法操纵结果,它会立即打印出来:
new Exception().printStackTrace();
两种解决方案都存在同样的问题——它们都急切地捕获整个堆栈的快照,可不方便使用。
2. JEP-259: Stack-Walking API
JEP-259应该解决这些问题,而且确实如此。 新的API提供了一种使用Stream API懒惰地遍历堆栈跟踪的便捷方法。
我们可以像这样轻松地创建StackWalker 实例:
StackWalker stack = StackWalker.getInstance();
此外,我们可以提供一些初始选项:
StackWalker = StackWalker.getInstance(StackWalker.Option.RETAIN_CLASS_REFERENCE);
如果我们想要遍历整个堆栈,那只需要调用forEach()
方法:
stack.forEach(System.out::println);
3. StackWalker.StackFrame
如果我们查看Java 1.4的StackTraceElement——它几乎是一个包含有关声明类、方法名、类加载器名等的详细字符串信息。
StackWalker.StackFrame是一个更加类型安全友好的升级,在其上面提到了丰富的方法:
public Class<?> getDeclaringClass();
public MethodType getMethodType();
…甚至可这样:
public StackTraceElement toStackTraceElement();
4.示例
让我们将前面那些付诸实践,来创建一个简单的调用层次结构
(代码包和类名: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运行显示会不一样,这个它的处理模式有关)中运行它,结果将是(注意堆栈元素的顺序):
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 <T> T walk(Function<? super Stream<StackFrame>, ? extends T> function)
还有一个很好的理由使它成为这种方式——堆栈需要被冻结以便遍历它,并且这发生在walk()方法调用的范围内 - 所以使用基于函数接口的模板方法实现这一目标是有意义的 。
即使你试图通过返回一个Stream实例来欺骗它,它也无法使用(自己试试看!)。
一旦我们知道了这个限制,我们只受我们的想象力和Stream API功能的约束。例如,我们可以优雅地跳过一些帧,然后挑选第一个遇到的帧:
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);
// 结果如下:
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);
}
}
总结
以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,谢谢大家对脚本之家的支持。如果你想了解更多相关内容请查看下面相关链接
来源:https://www.toutiao.com/a6600896038721028622/


猜你喜欢
- 准备数据data class ContactEntity( val letter: Char, &n
- 一、创建maven项目我使用的是汉化的idea可以选择原型,我这里没有选择输入项目名称,完成创建二、配置tomcat选择运行编辑配置点加号找
- 网页爬虫:其实就是一个程序用于在互联网中获取符合指定规则的数据。package day05; import java.io.Buffered
- 上一篇文章Android进程间通信(IPC)机制Binder简要介绍和学习计划简要介绍了Android系统进程间通
- 在安装过后出现了这样的问题:于是看了一下,是找不到这个版本,于是到gradle文件里加了一句话,指定好版本,切记不要低于26,然后去sdk
- 本文实例为大家分享了unity鼠标或者手指点击模型播放动的具体代码,供大家参考,具体内容如下using UnityEngine;using
- 操作字符串的类都有哪些?区别是什么?操作字符串的类主要用三个,分别是String类,StringBuffer类和StringBuilder类
- 第一步:项目中资源配置文件夹(resources文件夹)下先新增测试环境application-dev.yml和application-pr
- /// <summary>/// 获取本机在局域网的IP地址/// </summary>/// <return
- 一、递归的思路一个方法在执行时,调用自身被称为“递归”。递归相当于数学归纳法,有一个起始条件,有一个递推公式。递归可以分为:单路递归和多路递
- 本文实例为大家分享了Android Studio实现登录界面的具体代码,供大家参考,具体内容如下题目设计一个登录界面。要求:a) 包含用户名
- Java中避免NullPointerException的方法总结在字符串常量上调用equals// good"string lit
- 正文前: 1. IDEA内存优化(秒开的快感!!)因机器本身的配置而配置:\IntelliJ IDEA8\bin\idea.exe.vmop
- 在Spring中有一个类CachingUserDetailsService实现了UserDetailsService接口,该类使用静态代理模
- 背景spring的profile大家都是用的溜的飞起~那么profile的组合如何使用呢???比如我们这样使用@Profile({"
- jdk * 和cglib * 实现及区别代理模式是一种设计模式,提供了对目标对象额外的访问方式,即通过代理对象访问目标对象,这样可以在不
- import java.io.FileNotFoundException;import java.io.FileOutputStream;i
- 本文实例为大家分享了Springboot整合pagehelper分页展示的具体代码,供大家参考,具体内容如下一、添加依赖查找maven中pa
- <script>//验证身份证号方法var test=function(idcard){var Errors=new Array
- SpringDataJpa创建中间表//fetch=FetchType.EAGER 关闭懒加载 相当于hibernate中的lazy=fal