一、什么是堆栈信息
在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 <0x0000000795762ee8> (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 <0x0000000795760c08> (a java.io.OutputStreamWriter) at java.io.OutputStreamWriter.flush(OutputStreamWriter.java:229) at java.util.logging.StreamHandler.flush(StreamHandler.java:242) - locked <0x00000007957d91c0> (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 <0x0000000703346e40> (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堆栈信息分析是程序调试和优化的重要手段。通过分析堆栈信息,可以快速定位线程死锁、性能瓶颈和异常等问题,并针对性地进行优化。