目录
一、原理分析
(一)介绍
(二)IDEA和eclipse 调试原理为
(三)架构体系
二、远程调试实例
三、JDI工具代码实践
(一)JDI技术架构
(二)实践案例
首页 Java java教程 Java平台调试体系原理是什么

Java平台调试体系原理是什么

May 21, 2023 pm 11:47 PM
java

    一、原理分析

    (一)介绍

    JPDA(Java Platform Debugger Architecture) 是 Java 平台调试体系结构的缩写,通过 JPDA 提供的 API,开发人员可以方便灵活的搭建 Java 调试应用程序。

    JPDA 主要由三个部分组成:Java 虚拟机工具接口(JVMTI),Java 调试线协议(JDWP),以及 Java 调试接口(JDI)。

    Java 程序都是运行在 Java 虚拟机上的,我们要调试 Java 程序,事实上就需要向 Java 虚拟机请求当前运行态的状态,并对虚拟机发出一定的指令,设置一些回调等等,那么 Java 的调试体系,就是虚拟机的一整套用于调试的工具和接口。

    (二)IDEA和eclipse 调试原理为

    1:编辑器作为客户端和服务器程序通过暴露的监听端口建立socket连接

    2:IDE客户端将断点位置创建了断点事件通过 JDI 接口传给了 服务端(程序端)的 VM,VM 调用 suspend 将 VM 挂起

    3:VM 挂起之后将客户端需要获取的 VM 信息返回给客户端,返回之后 VM resume 恢复其运行状态

    4:客户端获取到 VM 返回的信息之后可以通过不同的方式进行展示

    (三)架构体系

    JPDA 定义了一个完整独立的体系,它由三个相对独立的层次共同组成,而且规定了它们三者之间的交互方式,或者说定义了它们通信的接口。

    这三个层次由低到高分别是 Java 虚拟机工具接口(JVMTI),Java 调试线协议(JDWP)以及 Java 调试接口(JDI)。

    这三个模块把调试过程分解成几个很自然的概念:调试者(debugger)和被调试者(debuggee),以及他们中间的通信器。

    被调试者运行于我们想调试的 Java 虚拟机之上,它可以通过 JVMTI 这个标准接口,监控当前虚拟机的信息;调试者定义了用户可使用的调试接口,通过这些接口,用户可以对被调试虚拟机发送调试命令,同时调试者接受并显示调试结果。

    JDWP通讯协议被用于传输调试命令和调试结果,在进行调试时它连接了调试者和被调试者。所有的命令被封装成 JDWP 命令包,通过传输层发送给被调试者,被调试者接收到 JDWP 命令包后,解析这个命令并转化为 JVMTI 的调用,在被调试者上运行。

    相似的情况是,JVMTI 通过将运行结果转化成 JDWP 数据包的形式,将结果发往调试者并且回传给 JDI 调用。而调试器开发人员就是通过 JDI 得到数据,发出指令。

    Java平台调试体系原理是什么

    如上图所示JPDA 由三层组成:

    • JVM TI - Java VM 工具接口。定义 VM 提供的调试服务。

    • JDWP - Java 调试通信协议。定义被调试者和调试器进程之间的通信。

    • JDI - Java 调试接口。定义一个高级 Java 语言接口,工具开发人员可以轻松地使用它来编写远程调试器应用程序。

    通过 JPDA 这套接口,我们就可以开发自己的调试工具。通过这些 JPDA 提供的接口和协议,调试器开发人员就能根据特定开发者的需求,扩展定制 Java 调试应用程序。

    前面我们提到的 IDE 调试工具都是基于 JPDA 体系开发的,区别仅仅在于它们可能提供了不同的图形界面、具有一些不同的自定义功能。

    另外,我们要注意的是,JPDA 是一套标准,任何的 JDK 实现都必须完成这个标准,因此,通过 JPDA 开发出来的调试工具先天具有跨平台、不依赖虚拟机实现、JDK 版本无关等移植优点,因此大部分的调试工具都是基于这个体系的。

    二、远程调试实例

    【1】构建一个SpringBoot的WEB项目。我们目前正在使用的SpringBoot版本为2.3.0.RELEASE。对应的tomcat版本是9.X。

    将该SpringBoot项目进行打包,并将应用程序的端口设置为9999。部署此程序至Linux服务器,无论采用JAR包还是Docker方式,与远程调试无关。

    【3】部署程序的代码参考如下,就是一个简单的请求处理打印输出信息

    /** 
     *   测试程序
     * @author zhangyu
     * @date 2022/2/17
    */
    @SpringBootApplication
    @RestController
    public class DebuggerApplication {
    
        public static void main(String[] args) {
            SpringApplication.run(DebuggerApplication.class, args);
        }
    
        @GetMapping("/test")
        public  String test(){
            System.out.println(111);
            System.out.println(222);
            return "OK";
        }
    
    }
    登录后复制

    【4】部署程序启动参数如下

    java -Xdebug -Xrunjdwp:transport=dt_socket,server=y,suspend=n,address=8888 -jar debugger-0.0.1-SNAPSHOT.jar
    登录后复制

    其中address=8888表示开启8888端口作为远程调试的Socket通信端口

    Java平台调试体系原理是什么

    如果是部署在tomcat下的普通web项目,参考如下:

    小于 tomcat9 版本

    tomcat 中 bin/catalina.sh 中增加 CATALINA_OPTS=‘-server -Xdebug -Xnoagent -Djava.compiler=NONE -Xrunjdwp:transport=dt_socket,server=y,suspend=n,address=18006’

    如下图所示:

    Java平台调试体系原理是什么

    大于等于 tomcat9 版本

    tomcat 中 bin/catalina.sh 中的 JPDA_ADDRESS=“localhost:8000” 这一句中的localhost修改为0.0.0.0(允许所有ip连接到8000端口,而不仅是本地)8000是端口,端口号可以任意修改成没有占用的即可

    如下图所示:

    Java平台调试体系原理是什么

    【5】测试部署的程序正常后,下面构建客户端远程调试,当前以IDEA工具作为客户端

    Java平台调试体系原理是什么

    参考:

    【1】-agentlib:jdwp=transport=dt_socket,server=y,suspend=n,address=8888

    【2】Host:远程服务器地址

    【3】Port:远程服务器开放的调试通信端口,非应用端口

    Java平台调试体系原理是什么

    Java平台调试体系原理是什么

    测试接口:http://XXX:9999/test。注意本地代码需要和远程部署程序一致。

    通过上图可以看到客户端设置断点已经生效,其中在客户端执行了一个调试输出,这个是自定义输出的内容服务器程序并没有,在执行后右侧的服务器控制台日志输出了该信息,因此远程Debug是正常通信和处理的。

    (一)调试参数详解

    • -Xdebug:启用调试特性

    • -Xrunjdwp:在目标 VM 中加载 JDWP 实现。它通过传输和 JDWP 协议与独立的调试器应用程序通信。下面介绍一些特定的子选项

    从 Java V5 开始,您可以使用 -agentlib:jdwp 选项,而不是 -Xdebug 和 -Xrunjdwp。如果连接到VM版本低于V5的情况下,只能使用 -Xdebug 和 -Xrunjdwp 选项。下面简单描述 -Xrunjdwp 子选项。

    • -Djava.compiler=NONE: 禁止 JIT 编译器的加载

    • transport : 传输方式,有 socket 和 shared memory 两种,我们通常使用 socket(套接字)传输,但是在 Windows 平台上也可以使用shared memory(共享内存)传输。

    • server(y/n): VM 是否需要作为调试服务器执行

    • address: 调试服务器的端口号,客户端用来连接服务器的端口号

    • suspend(y/n):值是 y 或者 n,若为 y,启动时候自己程序的 VM 将会暂停(挂起),直到客户端进行连接,若为 n,自己程序的 VM 不会挂起

    三、JDI工具代码实践

    (一)JDI技术架构

    Java平台调试体系原理是什么

    (二)实践案例

    (1)被调试程序

    创建一个SpringBoot的WEB项目,提供一个简单的测试接口,并在测试方法中提供一些方法参数变量和局部变量作为后面的调试测试用。

    import org.springframework.boot.SpringApplication;
    import org.springframework.boot.autoconfigure.SpringBootApplication;
    import org.springframework.web.bind.annotation.GetMapping;
    import org.springframework.web.bind.annotation.RestController;
    
    @SpringBootApplication
    @RestController
    public class DebuggerApplication {
    
        public static void main(String[] args) {
            SpringApplication.run(DebuggerApplication.class, args);
        }
    
        @GetMapping("/test")
        public  String test(String name){
            System.out.println("进入方法");
            int var=100;
            System.out.println(name);
            System.out.println(var);
            System.out.println("方法结束");
            return "OK";
        }
    }
    登录后复制

    项目启动配置参考,需要启用Debug配置

    Java平台调试体系原理是什么

    (2)自定义调试器代码

    开发调试器需要JNI工具支持,JDI操作的API工具在tools.jar中 ,需要在 CLASSPATH 中添加/lib/tools.jar

    Java平台调试体系原理是什么

    import com.sun.jdi.*;
    import com.sun.jdi.connect.AttachingConnector;
    import com.sun.jdi.connect.Connector;
    import com.sun.jdi.event.*;
    import com.sun.jdi.request.BreakpointRequest;
    import com.sun.jdi.request.EventRequest;
    import com.sun.jdi.request.EventRequestManager;
    import com.sun.tools.jdi.SocketAttachingConnector;
    
    import java.util.List;
    import java.util.Map;
    
    /**
     *  通过JNI工具测试Debug
     * @author zhangyu
     * @date 2022/2/20
    */
    public class TestDebugVirtualMachine {
    
    
        private static VirtualMachine vm;
    
        public static void main(String[] args) throws Exception {
            //获取SocketAttachingConnector,连接其它JVM称之为附加(attach)操作
            VirtualMachineManager vmm = Bootstrap.virtualMachineManager();
            List<AttachingConnector> connectors = vmm.attachingConnectors();
            SocketAttachingConnector sac = null;
            for(AttachingConnector ac : connectors) {
                if(ac instanceof SocketAttachingConnector) {
                    sac = (SocketAttachingConnector) ac;
                }
            }
            assert sac != null;
            //设置好主机地址,端口信息
            Map<String, Connector.Argument> arguments = sac.defaultArguments();
            Connector.Argument hostArg = arguments.get("hostname");
            Connector.Argument portArg = arguments.get("port");
            hostArg.setValue("127.0.0.1");
            portArg.setValue(String.valueOf(8800));
            //进行连接
             vm = sac.attach(arguments);
            //相应的请求调用通过requestManager来完成
            EventRequestManager eventRequestManager = vm.eventRequestManager();
            //创建一个代码判断,因此需要获取相应的类,以及具体的断点位置,即相应的代码行。
            ClassType clazz = (ClassType) vm.classesByName("com.zy.debugger.DebuggerApplication").get(0);
            //设置断点代码位置
            Location location = clazz.locationsOfLine(22).get(0);
            //创建新断点并设置阻塞模式为线程阻塞,即只有当前线程被阻塞
            BreakpointRequest breakpointRequest = eventRequestManager.createBreakpointRequest(location);
            //设置阻塞并启动
            breakpointRequest.setSuspendPolicy(EventRequest.SUSPEND_EVENT_THREAD);
            breakpointRequest.enable();
            //获取vm的事件队列
            EventQueue eventQueue = vm.eventQueue();
            while(true) {
                //不断地读取事件并处理断点记录事件
                EventSet eventSet = eventQueue.remove();
                EventIterator eventIterator = eventSet.eventIterator();
                while(eventIterator.hasNext()) {
                    Event event = eventIterator.next();
                    execute(event);
                }
                //将相应线程resume,表示继续运行
                eventSet.resume();
            }
        }
    
        /**
         *  处理监听到事件
         * @author zhangyu
         * @date 2022/2/20
        */
        public static void execute(Event event) throws Exception {
            //获取的event为一个抽象的事件记录,可以通过类型判断转型为具体的事件,这里我们转型为BreakpointEvent,即断点记录,
            BreakpointEvent breakpointEvent = (BreakpointEvent) event;
            //并通过断点处的线程拿到线程帧,进而获取相应的变量信息,并打印记录。
            ThreadReference threadReference = breakpointEvent.thread();
            StackFrame stackFrame = threadReference.frame(0);
            List<LocalVariable> localVariables = stackFrame.visibleVariables();
            //输出当前线程栈帧保存的变量数据
            localVariables.forEach(t -> {
                Value value = stackFrame.getValue(t);
                System.out.println("local->" + value.type() + "," + value.getClass() + "," + value);
            });
        }
    
    }
    登录后复制

    (3)代码分析

    【1】通过Bootstrap.virtualMachineManager();获取连接器,客户端即通过相应的connector进行连接,配置服务器程序ip地址和端口,连接后获取对应服务器的VM信息。

    定位目标debug的类文件,可通过遍历获取的类集合并结合VirtualMachine获取类信息实现

    【3】对目标类代码特定位置设置并启用断点

    【4】记录断点信息,阻塞服务器线程,并根据对应事件获取相应的信息

    【5】执行event.resume释放断点,服务器程序继续运行

    (4)运行测试

    启动 SpringBoot 的 web 项目,也就是服务器程序。启动调试器代码时使用debug模式,并在该位置查看所获取的信息,同时避免直接释放断点。

    Java平台调试体系原理是什么

    【2】设置断点位置为DebuggerApplication类的第22行

    Java平台调试体系原理是什么

    【3】启动后测试该接口,可以发现服务器程序控制台打印了如下结果。第22行还没有执行。

    Java平台调试体系原理是什么

    【4】此时,在观察调试器程序。可以看到获取到了服务器程序栈帧的数据

    Java平台调试体系原理是什么

    【5】释放断点,服务器正常运行完本次请求处理流程

    Java平台调试体系原理是什么

    以上是Java平台调试体系原理是什么的详细内容。更多信息请关注PHP中文网其他相关文章!

    本站声明
    本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

    热AI工具

    Undresser.AI Undress

    Undresser.AI Undress

    人工智能驱动的应用程序,用于创建逼真的裸体照片

    AI Clothes Remover

    AI Clothes Remover

    用于从照片中去除衣服的在线人工智能工具。

    Undress AI Tool

    Undress AI Tool

    免费脱衣服图片

    Clothoff.io

    Clothoff.io

    AI脱衣机

    Video Face Swap

    Video Face Swap

    使用我们完全免费的人工智能换脸工具轻松在任何视频中换脸!

    热工具

    记事本++7.3.1

    记事本++7.3.1

    好用且免费的代码编辑器

    SublimeText3汉化版

    SublimeText3汉化版

    中文版,非常好用

    禅工作室 13.0.1

    禅工作室 13.0.1

    功能强大的PHP集成开发环境

    Dreamweaver CS6

    Dreamweaver CS6

    视觉化网页开发工具

    SublimeText3 Mac版

    SublimeText3 Mac版

    神级代码编辑软件(SublimeText3)

    热门话题

    Java教程
    1663
    14
    CakePHP 教程
    1420
    52
    Laravel 教程
    1313
    25
    PHP教程
    1266
    29
    C# 教程
    1238
    24
    突破或从Java 8流返回? 突破或从Java 8流返回? Feb 07, 2025 pm 12:09 PM

    Java 8引入了Stream API,提供了一种强大且表达力丰富的处理数据集合的方式。然而,使用Stream时,一个常见问题是:如何从forEach操作中中断或返回? 传统循环允许提前中断或返回,但Stream的forEach方法并不直接支持这种方式。本文将解释原因,并探讨在Stream处理系统中实现提前终止的替代方法。 延伸阅读: Java Stream API改进 理解Stream forEach forEach方法是一个终端操作,它对Stream中的每个元素执行一个操作。它的设计意图是处

    PHP:网络开发的关键语言 PHP:网络开发的关键语言 Apr 13, 2025 am 12:08 AM

    PHP是一种广泛应用于服务器端的脚本语言,特别适合web开发。1.PHP可以嵌入HTML,处理HTTP请求和响应,支持多种数据库。2.PHP用于生成动态网页内容,处理表单数据,访问数据库等,具有强大的社区支持和开源资源。3.PHP是解释型语言,执行过程包括词法分析、语法分析、编译和执行。4.PHP可以与MySQL结合用于用户注册系统等高级应用。5.调试PHP时,可使用error_reporting()和var_dump()等函数。6.优化PHP代码可通过缓存机制、优化数据库查询和使用内置函数。7

    PHP与Python:了解差异 PHP与Python:了解差异 Apr 11, 2025 am 12:15 AM

    PHP和Python各有优势,选择应基于项目需求。1.PHP适合web开发,语法简单,执行效率高。2.Python适用于数据科学和机器学习,语法简洁,库丰富。

    PHP与其他语言:比较 PHP与其他语言:比较 Apr 13, 2025 am 12:19 AM

    PHP适合web开发,特别是在快速开发和处理动态内容方面表现出色,但不擅长数据科学和企业级应用。与Python相比,PHP在web开发中更具优势,但在数据科学领域不如Python;与Java相比,PHP在企业级应用中表现较差,但在web开发中更灵活;与JavaScript相比,PHP在后端开发中更简洁,但在前端开发中不如JavaScript。

    PHP与Python:核心功能 PHP与Python:核心功能 Apr 13, 2025 am 12:16 AM

    PHP和Python各有优势,适合不同场景。1.PHP适用于web开发,提供内置web服务器和丰富函数库。2.Python适合数据科学和机器学习,语法简洁且有强大标准库。选择时应根据项目需求决定。

    Java程序查找胶囊的体积 Java程序查找胶囊的体积 Feb 07, 2025 am 11:37 AM

    胶囊是一种三维几何图形,由一个圆柱体和两端各一个半球体组成。胶囊的体积可以通过将圆柱体的体积和两端半球体的体积相加来计算。本教程将讨论如何使用不同的方法在Java中计算给定胶囊的体积。 胶囊体积公式 胶囊体积的公式如下: 胶囊体积 = 圆柱体体积 两个半球体体积 其中, r: 半球体的半径。 h: 圆柱体的高度(不包括半球体)。 例子 1 输入 半径 = 5 单位 高度 = 10 单位 输出 体积 = 1570.8 立方单位 解释 使用公式计算体积: 体积 = π × r2 × h (4

    PHP的影响:网络开发及以后 PHP的影响:网络开发及以后 Apr 18, 2025 am 12:10 AM

    PHPhassignificantlyimpactedwebdevelopmentandextendsbeyondit.1)ItpowersmajorplatformslikeWordPressandexcelsindatabaseinteractions.2)PHP'sadaptabilityallowsittoscaleforlargeapplicationsusingframeworkslikeLaravel.3)Beyondweb,PHPisusedincommand-linescrip

    PHP:许多网站的基础 PHP:许多网站的基础 Apr 13, 2025 am 12:07 AM

    PHP成为许多网站首选技术栈的原因包括其易用性、强大社区支持和广泛应用。1)易于学习和使用,适合初学者。2)拥有庞大的开发者社区,资源丰富。3)广泛应用于WordPress、Drupal等平台。4)与Web服务器紧密集成,简化开发部署。

    See all articles