一、什么是堆栈信息
在Java程序中,为了存储数据而分配的内存分为两类:堆(Heap)和栈(Stack)。
堆是Java运行时内存中的一块区域,用于存储对象及其实例变量。而栈也是内存中的一块区域,用于存储方法执行时的局部变量和操作数。在Java程序中,当一个方法被调用时,Java虚拟机会在栈中为该方法创建一个新的栈帧,然后将其放在栈的顶部。当方法执行完毕时,栈帧就会被弹出,方法的返回值也会被压入栈中。
堆栈信息指的是程序在运行时,产生的一些状态信息,包括栈帧的大小、离线时间、运行时间。通过对这些信息的分析,可以排查代码的执行过程中出现的问题。
二、如何获取堆栈信息
Java虚拟机提供了一些工具和API,用于获取堆栈信息,比如jstack、jconsole、jvisualvm等。
jstack用于打印出指定Java进程中每个线程的堆栈信息。打印出来的信息包含线程状态、锁信息、分配的对象、堆栈跟踪信息等。
public class Test {
public static void main(String[] args) {
while (true) {
System.out.println("Hello, World!");
}
}
}
使用jstack命令可以获取到以下堆栈信息:
"main" #1 prio=5 os_prio=0 tid=0x0000000002c8f800 nid=0x3880 runnable [0x00000000027ff000]
java.lang.Thread.State: RUNNABLE
at java.io.PrintStream.write(PrintStream.java:480)
- locked (a java.io.PrintStream)
at sun.nio.cs.StreamEncoder.writeBytes(StreamEncoder.java:221)
at sun.nio.cs.StreamEncoder.implFlushBuffer(StreamEncoder.java:291)
at sun.nio.cs.StreamEncoder.implFlush(StreamEncoder.java:295)
at sun.nio.cs.StreamEncoder.flush(StreamEncoder.java:141)
- locked (a java.io.OutputStreamWriter)
at java.io.OutputStreamWriter.flush(OutputStreamWriter.java:229)
at java.util.logging.StreamHandler.flush(StreamHandler.java:242)
- locked (a java.util.logging.ConsoleHandler)
at java.util.logging.ConsoleHandler.publish(ConsoleHandler.java:106)
at java.util.logging.Logger.log(Logger.java:738)
at java.util.logging.Logger.doLog(Logger.java:765)
at java.util.logging.Logger.log(Logger.java:788)
at java.util.logging.Logger.info(Logger.java:1495)
at com.intellij.rt.execution.application.AppMain.main(AppMain.java:147)
三、堆栈信息分析的常用场景
1.线程死锁
线程死锁指的是两个或多个线程互相持有对方所需要的锁,从而导致所有线程都阻塞无法继续执行。通过分析堆栈信息,可以定位哪些线程被阻塞在了哪些锁上,从而帮助我们找到死锁的根源并解决问题。
public class Test {
public static final Object lock1 = new Object();
public static final Object lock2 = new Object();
public static void main(String[] args) {
new Thread(new Runnable() {
@Override
public void run() {
synchronized (lock1) {
System.out.println("Thread 1: Holding lock 1...");
try {
Thread.sleep(10);
} catch (InterruptedException e) {
}
System.out.println("Thread 1: Waiting for lock 2...");
synchronized (lock2) {
System.out.println("Thread 1: Holding lock 1 and 2...");
}
}
}
}).start();
new Thread(new Runnable() {
@Override
public void run() {
synchronized (lock2) {
System.out.println("Thread 2: Holding lock 2...");
try {
Thread.sleep(10);
} catch (InterruptedException e) {
}
System.out.println("Thread 2: Waiting for lock 1...");
synchronized (lock1) {
System.out.println("Thread 2: Holding lock 1 and 2...");
}
}
}
}).start();
}
}
使用jstack命令可以获取到以下堆栈信息:
"Thread-0": waiting to lock Monitor@0x0000000704003840 (Object@0x000000076c4a52a8, a java.lang.Object), which is held by "Thread-1" "Thread-1": waiting to lock Monitor@0x0000000704002b60 (Object@0x000000076c4a52b8, a java.lang.Object), which is held by "Thread-0"
从以上堆栈信息可以看出,Thread-0持有的锁(Object@0x000000076c4a52a8)正在被Thread-1等待获取,而Thread-1持有的锁(Object@0x000000076c4a52b8)正在被Thread-0等待获取。这就是典型的死锁场景。
2.性能分析
通过分析堆栈信息,可以了解到程序在执行过程中的性能瓶颈,从而针对性地进行优化。
public class Test {
public static void main(String[] args) {
long start = System.currentTimeMillis();
for (int i = 0; i < 10000000; i++) {
new String("hello");
}
long end = System.currentTimeMillis();
System.out.println("Time used: " + (end - start) + " ms.");
}
}
使用jstack命令可以获取到以下堆栈信息:
"main" #1 prio=5 os_prio=0 tid=0x00000000022ce800 nid=0x34b8 runnable [0x000000000215f000]
java.lang.Thread.State: RUNNABLE
at java.lang.String.<init>(String.java:604)
at com.example.Test.main(Test.java:7)
从以上堆栈信息可以看出,程序主要的耗时都花在了创建String对象上。因为每次创建String对象都需要在堆中分配内存,所以这部分操作非常消耗性能。如果需要频繁地创建相同的字符串,可以考虑使用字符串常量池来避免重复创建对象。
3.异常排查
通过分析堆栈信息,可以快速定位程序运行过程中的异常。
public class Test {
public static void main(String[] args) {
try {
System.out.println(3 / 0);
} catch (Exception e) {
e.printStackTrace();
}
}
}
使用jstack命令可以获取到以下堆栈信息:
"main" #1 prio=5 os_prio=0 tid=0x0000000002e61000 nid=0x2370 waiting on condition [0x00000000027fc000]
java.lang.Thread.State: WAITING (parking)
at sun.misc.Unsafe.park(Native Method)
- parking to wait for (a java.util.concurrent.locks.AbstractQueuedSynchronizer$ConditionObject)
at java.util.concurrent.locks.LockSupport.park(LockSupport.java:175)
at java.util.concurrent.locks.AbstractQueuedSynchronizer$ConditionObject.await(AbstractQueuedSynchronizer.java:2039)
at java.util.concurrent.LinkedBlockingQueue.take(LinkedBlockingQueue.java:442)
at java.util.concurrent.ThreadPoolExecutor.getTask(ThreadPoolExecutor.java:1074)
at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1134)
at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:624)
at java.lang.Thread.run(Thread.java:748)
从以上堆栈信息可以看出,程序在执行3/0操作时抛出了异常。经过分析,发现异常是由于除数为0造成的。
四、结论
Java堆栈信息分析是程序调试和优化的重要手段。通过分析堆栈信息,可以快速定位线程死锁、性能瓶颈和异常等问题,并针对性地进行优化。
原创文章,作者:小蓝,如若转载,请注明出处:https://www.506064.com/n/303855.html
微信扫一扫
支付宝扫一扫