为什么我的对象被IntelliJ IDEA的调试器无声地更改了?

问题描述 投票:5回答:1

我正在调试(Shift + F9)JDK8 API的offer(E e)方法java.util.concurrent.ConcurrentLinkedQueue

我发现IntelliJ IDEA会在调试过程中无声地更改此队列的head字段,

但是运行模式(Shift + F10)不会改变字段,为什么?

并且在offer(E e)中没有代码更改此字段,这让我感到困惑。

然后,我尝试了另一个IDE(Eclipse Mars.1 Release(4.5.1)),它没有这样的问题。

我的测试有一些结果:

  • 运行模式(Shift + F10)
head: 1159190947 
head: 1159190947 
head: 1159190947 
head: 1159190947 
  • 调试架构(跳过:F8)
head: 1989972246 
head: 1791930789 
head: 1791930789 
head: 1791930789 
  • 调试模式(简历程序:F9)
head: 1989972246 
head: 1989972246 
head: 1791930789 
head: 1791930789 

版本信息 : IntelliJ IDEA 2018.1(终极版) Build#IU-181.4203.550,建于2018年3月27日 JRE:1.8.0_162-b12 amd64 JVM:Oracle Corporation的Java HotSpot(TM)64位服务器VM Windows 10 10.0

IntelliJ IDEA 2018.2.3(社区版)和IntelliJ IDEA 2019.1(终极版)也存在同样的问题。

import java.lang.reflect.Field;
import java.util.concurrent.ConcurrentLinkedQueue;

public class ConcurrentLinkedQueueTest {

    public static void main(String[] args) {
        ConcurrentLinkedQueue<String> queue = new ConcurrentLinkedQueue<>();
        print(queue);
        queue.offer("aaa");
        print(queue);
        queue.offer("bbb");
        print(queue);
        queue.offer("ccc");
        print(queue);
    }

    /**
     * 打印并发队列head属性的identityHashCode
     * @param queue
     */
    private static void print(ConcurrentLinkedQueue queue) {
        Field field = null;
        boolean isAccessible = false;
        try {
            field = ConcurrentLinkedQueue.class.getDeclaredField("head");
            isAccessible = field.isAccessible();
            if (!isAccessible) {
                field.setAccessible(true);
            }
            System.out.println("head: " + System.identityHashCode(field.get(queue)));
        } catch (Exception e) {
            e.printStackTrace();
        } finally {
            field.setAccessible(isAccessible);
        }
    }
}
java debugging intellij-idea
1个回答
5
投票

默认情况下,IntelliJ IDEA在对象上调用toString()方法以在Watches视图中获取它们的表示。在ConcurrentLinkedQueue中,toString()方法迭代集合,它可以通过head方法中的updateHead()调用更新first()字段(并且可能在其他地方,我没有调查整个实现。

如果你去首选项|构建,执行,部署|调试器|数据视图| Java并关闭“启用toString()对象视图”,您不应再观察此行为。

© www.soinside.com 2019 - 2024. All rights reserved.