IntelliJ IDEA遠程Debug Linux的Java程序,找問題不要只會看日志了


1 前言

歡迎訪問南瓜慢說 www.pkslow.com獲取更多精彩文章!

我們習慣於在本地開發的時候debug,能快速定位與解決問題,那部署在服務器上是不是就沒有辦法了呢?只能通過查看日志來定位?

不是的,在遠端的服務器上,我們一樣可以debug

2 IDEA的debug

我們先來看一下在IntelliJ IDEA直接debug是怎樣的。

先准備一個簡單的Java程序:

package com.pkslow.basic;
import java.util.Map;

public class RemoteDebug {
    public static void main(String[] args) {
        System.out.println("------------------start------------------");
        
        System.out.println("get all the system environment");
        Map<String, String> envs = System.getenv();

        System.out.println("\nprint out the contains `HOME`");
        System.out.println("------env HOME------");
        envs.entrySet().stream()
                .filter(env -> env.getKey().contains("HOME"))
                .forEach(env -> {
                    System.out.println(env.getKey() + ":" + env.getValue());
                });

        System.out.println("------------------end------------------");
    }
}

功能很簡單,獲取所有系統環境變量,並打印出含有HOME字段的。

Debug很簡單,直接點擊以下按鈕就可以:

相信大家都知道這一點,但應該很多人都不會注意,IDEA究竟做了什么,為什么就可以調試了呢?我們看一下控制台的日志就明白了:

/Library/Java/JavaVirtualMachines/jdk1.8.0_212.jdk/Contents/Home/bin/java -agentlib:jdwp=transport=dt_socket,address=127.0.0.1:59313,suspend=y,server=n -javaagent:/Users/pkslow/Library/Caches/IntelliJIdea2019.3/captureAgent/debugger-agent.jar -Dfile.encoding=UTF-8 -classpath "/Library/Java/JavaVirtualMachines/jdk1.8.0_212.jdk/Contents/Home/jre/lib/charsets.jar:" com.pkslow.basic.RemoteDebug
Connected to the target VM, address: '127.0.0.1:59313', transport: 'socket'

簡化一下,不重要的參數去掉:

java -agentlib:jdwp=transport=dt_socket,address=127.0.0.1:59313,suspend=y,server=n com.pkslow.basic.RemoteDebug

這就是可以Debug的原因,利用了Java Agent原理。這個功能很強大,類似一個AOP,代理了Java程序,可以利用它進行調試、熱部署等。

3 調試本地程序

我們先試試如何可以調試本地程序,不是直接在IDEA上調試。先要編譯出class文件RemoteDebug.class,然后按package結構放好。我通過mvn clean compile來編譯。

啟動程序,在target/classes/目錄執行:

$ java -agentlib:jdwp=transport=dt_socket,address=127.0.0.1:50050,suspend=y,server=y com.pkslow.basic.RemoteDebug
Listening for transport dt_socket at address: 50050

然后程序就會等待調試客戶端的連接,不會往下執行。

配置IDEA以進行調試:

配置完成保存后,點擊debug就可以了:

程序已經進入debug模式:

我們已經執行到了其中一行,現在看看服務端:

IDEA是同步的,並且確實已經控制了服務端Java的執行。

4 遠程調試Linux Java程序

先把程序部署在Linux上:

$ scp -P 22 ./com/pkslow/basic/RemoteDebug.class root@xxx.xx.xx.xxx:/root/remoteDebug/com/pkslow/basic/
RemoteDebug.class                              100% 2572   282.5KB/s   00:00    

通過以下命令在服務端啟動程序,這里調試端口改為9999,因為部分端口在遠程服務器並沒有開啟:

java -agentlib:jdwp=transport=dt_socket,address=9999,suspend=y,server=y com.pkslow.basic.RemoteDebug

本地電腦IDEA配置如下:

開始debug,正常控制了遠程服務端的Java

服務端的實時執行情況:

讓程序執行完如下:

$ java -agentlib:jdwp=transport=dt_socket,address=9999,suspend=y,server=y com.pkslow.basic.RemoteDebug
Listening for transport dt_socket at address: 9999
------------------start------------------
get all the system environment

print out the contains `HOME`
------env HOME------
JAVA_HOME:/root/jdk1.8.0_131
HOME:/root
------------------end------------------

5 總結

本文一步步探索如何進行調試遠程的服務器,這在出現問題時定位還是非常有用的。畢竟可以實時看到服務端運行環境。


歡迎關注微信公眾號<南瓜慢說>,將持續為你更新...

多讀書,多分享;多寫作,多整理。


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM