我正在调试(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)),它没有这样的问题。
我的测试有一些结果:
head: 1159190947
head: 1159190947
head: 1159190947
head: 1159190947
head: 1989972246
head: 1791930789
head: 1791930789
head: 1791930789
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);
}
}
}
默认情况下,IntelliJ IDEA在对象上调用toString()
方法以在Watches视图中获取它们的表示。在ConcurrentLinkedQueue
中,toString()
方法迭代集合,它可以通过head
方法中的updateHead()
调用更新first()
字段(并且可能在其他地方,我没有调查整个实现。
如果你去首选项|构建,执行,部署|调试器|数据视图| Java并关闭“启用toString()对象视图”,您不应再观察此行为。