乔志勇的博客

个人小站

Idea Intellij多线程调试

(适用Idea 14 & 15,其他版本未知)

几天前遇到一个问题: 
在多线程调试的时候,一些断点会被跳过。

比如像下面的代码:

public static void main(String[] args) throws InterruptedException {
    new Thread() { // 断点0
        @Override
        public void run() {
            System.out.println("1"); // 断点1
            try {
                Thread.sleep(2000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println("2"); // 断点2
        }
    }.start();
    // 外线程
    System.out.println("3"); // 断点3
    Thread.sleep(2000);
    System.out.println("4"); // 断点4
}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18

启动调试后,可能会命中断点1或者断点3,命中路径可以是3->2->4或者3->4甚至3->2,总之并不会命中全部断点。在单步调试的过程中,代码的执行速度事实上远比直接运行慢得多,所以我加入了一些sleep用来模拟这些延迟的过程。

这段代码一共有两个线程,在调试一个线程的过程中,很显然另一个线程是在运行的。所以我们可以猜测是不是Idea是不是会在我们调试一个线程时,屏蔽另一个线程的断点。 
为了验证这个观点,只需要在命中断点0处之后单步,就会到断点3,然会你调试得“慢”一点,就会发现断点1和2被跳过了。

这里写图片描述 
打钩的是我当前调试的线程(main线程),当你开始单步调试之后,这个堆栈会刷新。

这显然是不能接受的,对于多线程调试来说,最重要的就是控制两个线程的执行顺序,我们要做的其实就是当另一个不处于调试状态的线程命中断点后,能先暂停,一直等到我去处理为止。

经过一番折腾后发现,其实Idea提供了这个功能,在断点处右键 
这里写图片描述 
它提供了两种挂起的模式,默认的是All,只需要选中Thread,它就会一直等待到你处理它。 
右边的Make Default功能会使得之后打上的断点也会是Thread模式的(注意,之前打上的不会变更,需要手工更改)。 
这里写图片描述 
两个红圈的按钮就可以改,在Debug窗口的最右边。

之后就可以开心地调试了,对了,在这里切换线程 
这里写图片描述

阅读更多
个人分类: idea使用
上一篇Docker镜像上传到阿里云的步骤详解
下一篇PostgreSQL的MVCC并发处理
想对作者说点什么? 我来说一句

intellij IDEA使用教程

2017年09月01日 33.44MB 下载

IntelliJ IDEA 中文文档 __

2018年01月23日 20.8MB 下载

没有更多推荐了,返回首页

关闭
关闭