JStack实现浅析

相信Java程序员对jstack都不陌生,今天我们坑进OpenJDK源码来看看jstack是怎么实现的。
通过官方文档的介绍,我们可以知道jstack的源码在OpenJDK的jdk子项目下,sun.tools.jstack.JStack

/*
 * This class is the main class for the JStack utility. It parses its arguments
 * and decides if the command should be executed by the SA JStack tool or by
 * obtained the thread dump from a target process using the VM attach mechanism
 */
public class JStack { ... }

通过注释我们可以知道,jstack会根据不同参数决定使用不同的方式来进行thread dump:使用SA的JStack,或者使用attach机制。
通过代码可以看到,下面三种情况会使用SA的JStack,

  • 使用了-m选项;
  • 使用了-F选项;
  • 不是对本地进程而是对远程server执行jstack;

下面我们来看下使用attach机制实现的thread dump(SA的实现就先欠着啦:)。
关于attach机制,暴露给用户的其实就是Attach API,其底层实现,简单来讲就是在虚拟机启动了一个Attach Listener,外部进程通过UNIX socket与其进行通信,发送一些命令让Attach Listener去执行。具体可以参考官方文档或者这一篇博文
下面我们来看下jstack是如何使用attach机制的。runThreadDump方法,

    // Attach to pid and perform a thread dump
    private static void runThreadDump(String pid, String args[]) throws Exception {
        VirtualMachine vm = null;
        try {
            ///
            // 使用 Attach API
            ///
            vm = VirtualMachine.attach(pid);
        } catch (Exception x) {
            String msg = x.getMessage();
            if (msg != null) {
                System.err.println(pid + ": " + msg);
            } else {
                x.printStackTrace();
            }
            if ((x instanceof AttachNotSupportedException) &&
                (loadSAClass() != null)) {
                System.err.println("The -F option can be used when the target " +
                    "process is not responding");
            }
            System.exit(1);
        }

        ///
        // 发送命令执行thread dump
        ///
        // Cast to HotSpotVirtualMachine as this is implementation specific
        // method.
        InputStream in = ((HotSpotVirtualMachine)vm).remoteDataDump((Object[])args);

        // read to EOF and just print output
        byte b[] = new byte[256];
        int n;
        do {
            n = in.read(b);
            if (n > 0) {
                String s = new String(b, 0, n, "UTF-8");
                System.out.print(s);
            }
        } while (n > 0);
        in.close();
        vm.detach();
    }

可以看到最后是通过HotSpotVirtualMachine#remoteDataDump来执行thread dump。来看下这个方法

    // Remote ctrl-break. The output of the ctrl-break actions can
    // be read from the input stream.
    public InputStream remoteDataDump(Object ... args) throws IOException {
        return executeCommand("threaddump", args);
    }
    private InputStream executeCommand(String cmd, Object ... args) throws IOException {
        try {
            return execute(cmd, args);
        } catch (AgentLoadException x) {
            throw new InternalError("Should not get here", x);
        }
    }

继续往下跟代码到Linux平台上的实现,LinuxVirtualMachine#execute,可以看到最终通过UNIX socket与Attach Listener进行了通信,发送的就是上面看到的threaddump命令。Attach Listener对接收到的命令所执行的操作,在attachListener.cpp

// names must be of length <= AttachOperation::name_length_max
static AttachOperationFunctionInfo funcs[] = {
  { "agentProperties",  get_agent_properties },
  { "datadump",         data_dump },
  { "dumpheap",         dump_heap },
  { "load",             JvmtiExport::load_agent_library },
  { "properties",       get_system_properties },
  { "threaddump",       thread_dump },
  { "inspectheap",      heap_inspection },
  { "setflag",          set_flag },
  { "printflag",        print_flag },
  { "jcmd",             jcmd },
  { NULL,               NULL }
};

看下threaddump所对应的thread_dump方法,

// Implementation of "threaddump" command - essentially a remote ctrl-break
// See also: ThreadDumpDCmd class
//
static jint thread_dump(AttachOperation* op, outputStream* out) {
  bool print_concurrent_locks = false;
  if (op->arg(0) != NULL && strcmp(op->arg(0), "-l") == 0) {
    print_concurrent_locks = true;
  }

  // thread stacks
  VM_PrintThreads op1(out, print_concurrent_locks);
  VMThread::execute(&op1);

  // JNI global handles
  VM_PrintJNI op2(out);
  VMThread::execute(&op2);

  // Deadlock detection
  VM_FindDeadlocks op3(out);
  VMThread::execute(&op3);

  return JNI_OK;
}

所以再往下就得坑进HotSpot的源码,VMThread::execute,才能看到最终HotSpot是怎么执行的thread dump了。这里就不继续挖坑了,感兴趣的同学可以继续跟下去 ^_^

嗯,今天就先到这,have fun la

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: Linux jstack是一个命令行工具,用于生成Java线程转储。它可以帮助开发人员分析Java应用程序中的线程问题,如死锁、死循环等。jstack命令可以显示Java虚拟机中所有线程的状态,包括线程ID、线程名称、线程状态、线程堆栈等信息。开发人员可以使用这些信息来诊断线程问题并进行调试。 ### 回答2: linux jstack是一个用于诊断和调试Java应用程序的实用程序。它提供了一种查看Java应用程序运行时线程状态的方式,可以帮助开发人员分析线程堆栈,定位和解决应用程序中的性能问题和死锁等并发问题。 使用jstack命令可以获取Java应用程序的线程信息和线程堆栈信息。当应用程序出现性能问题时,可以使用jstack命令来查看应用程序的线程状态,以确定是否存在线程阻塞、死锁或竞争条件等问题。 使用jstack命令非常简单,只需要在终端中输入"jstack"加上Java应用程序的进程ID即可。命令执行后,会输出每个线程的ID、状态、所属的进程ID、线程堆栈信息等。通过分析线程堆栈信息,可以找到应用程序中可能存在的性能瓶颈和并发问题。 jstack命令还可以与其他性能诊断工具和分析工具配合使用。例如,可以将jstack输出的线程堆栈信息与VisualVM等工具进行对比分析,以进一步深入了解应用程序的性能瓶颈和并发问题。 总之,linux jstack是一个非常有用的工具,可以帮助开发人员诊断和调试Java应用程序。通过使用jstack命令,开发人员可以获得关于线程状态和线程堆栈信息的详细分析,以解决应用程序的性能问题和并发问题。 ### 回答3: linux jstack是一个在Linux系统上运行的命令行工具,用于生成Java线程的堆栈跟踪信息。 在Linux系统上,Java应用程序通常由Java虚拟机(JVM)执行。当JVM执行时,它会创建多个线程来并发执行不同的任务。这些线程之间可能会相互依赖,或者可能会出现死锁或其他线程相关的问题。 当出现线程相关的问题时,我们可以使用linux jstack命令来获取Java线程的堆栈跟踪信息。通过查看线程的堆栈跟踪信息,我们可以获得线程在执行过程中的详细信息,例如线程的调用栈、锁的所有权信息等。 要使用linux jstack命令,我们首先需要找到正在执行的Java应用程序的进程ID。然后,在终端中执行以下命令: jstack <PID> 其中<PID>是Java应用程序的进程ID。执行该命令后,linux jstack将生成一个包含所有Java线程的堆栈跟踪信息的输出文件。 通过分析这个输出文件,我们可以获得关于Java线程状态、锁信息、线程调用栈等详细信息。例如,我们可以查看哪个线程持有了锁、哪些线程正在等待锁等。 总而言之,linux jstack是一个用于生成Java线程堆栈跟踪信息的命令行工具。它是一个非常有用的工具,可以帮助我们在诊断和解决Java应用程序中的线程相关问题时提供有价值的信息。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值