linux下strace命令跟踪java进程

     strace命令的使用,是看了马士兵老师讲的netty底层原理课程,里面举了一个例子,使用java程序编写一个简单的socket服务器,然后等待客户端连接。

    本身,这个程序很简单,但是通过strace命令跟踪,可以看到,jvm底层调用了linux系统内核的socket,出现了一系列bind,listen,accept等等命令。

    strace这个命令并不是linux自带的,需要安装,centos系列可以直接通过yum安装。

    下面直接看示例:

    SocketTest.java

import java.io.*;
import java.net.*;

public class SocketTest{
  public static void main(String[] args) throws Exception{
    ServerSocket server = new ServerSocket(8090);
    System.out.println("step1: new ServerSocket(8090)");
    while(true){
      Socket client = server.accept();
      System.out.println("step2:client\t"+client.getPort());
      new Thread(new Runnable(){
        Socket ss;
        public Runnable setSocket(Socket s){
          ss = s;
          return this;
        }
        public void run(){
          try{
            InputStream in = ss.getInputStream();
            BufferedReader reader = new BufferedReader(new InputStreamReader(in));
            while(true){
               System.out.println(reader.readLine());
            }
          }catch(Exception e){
            e.printStackTrace();
          }
        }
      }.setSocket(client)).start();
    }
    
  }
}

    接着,进行如下几步:

  1. 编译java文件
  2. 开启跟踪
  3. 查看跟踪日志
  4. 模拟客户端连接继续看日志 

    这个时候,目录下回生成很多out开头带进程ID的文件,这都是跟踪文件。

    关键字bind,listen出现了,因为这时候没有客户端连接,所以进程挂起,等待客户端连接。

    现在模拟客户端连接,当我们输入连接命令,其他两个终端窗口显示日志发生了变更。

    切回第一个终端窗口,就是开启跟踪命令的窗口,发现打印了终端连接端口信息47642。

    切回到查看跟踪日志的终端窗口,日志信息发生了很大变化,需要滑动到最前面,我们看到了accept,并且 显示终端连接信息。

     这个示例,可以很清楚的看到,java的socket连接程序,运行在jvm上,底层的jvm去调用系统内核命令,这些命令bind,listen,accept都是linux下socket操作,我们如果写过c程序的套接字程序,会发现,就是这些命令的拼装与组合。java与c/c++有着很紧密的联系。

猜你喜欢

转载自blog.csdn.net/feinifi/article/details/118757564