Java Swing - 如何判断什么代码在组件上调用 paint?

标签 java multithreading swing

在我的代码中某处调用了 paint,但堆栈跟踪完全不知道是什么和在哪里。它也是间歇性的,所以我知道这是一个线程问题,如果我放置一个断点,那么代码永远不会失败。那么,我该如何判断事件线程在何处或如何被调用以重绘组件呢?

这是堆栈跟踪:

Exception in thread "AWT-EventQueue-0" java.lang.ArrayIndexOutOfBoundsException: 4
    at javax.swing.plaf.basic.BasicTabbedPaneUI.paintTabArea(BasicTabbedPaneUI.java:834)
    at javax.swing.plaf.basic.BasicTabbedPaneUI.paint(BasicTabbedPaneUI.java:797)
    at javax.swing.plaf.ComponentUI.update(ComponentUI.java:161)
    at javax.swing.JComponent.paintComponent(JComponent.java:778)
    at javax.swing.JComponent.paint(JComponent.java:1054)
    at javax.swing.JComponent.paintChildren(JComponent.java:887)
    at javax.swing.JComponent.paint(JComponent.java:1063)
    at javax.swing.JComponent.paintToOffscreen(JComponent.java:5219)
    at javax.swing.RepaintManager$PaintManager.paintDoubleBuffered(RepaintManager.java:1529)
    at javax.swing.RepaintManager$PaintManager.paint(RepaintManager.java:1452)
    at javax.swing.RepaintManager.paint(RepaintManager.java:1249)
    at javax.swing.JComponent._paintImmediately(JComponent.java:5167)
    at javax.swing.JComponent.paintImmediately(JComponent.java:4978)
    at javax.swing.RepaintManager$3.run(RepaintManager.java:808)
    at javax.swing.RepaintManager$3.run(RepaintManager.java:796)
    at java.security.AccessController.doPrivileged(Native Method)
    at java.security.ProtectionDomain$1.doIntersectionPrivilege(ProtectionDomain.java:76)
    at javax.swing.RepaintManager.paintDirtyRegions(RepaintManager.java:796)
    at javax.swing.RepaintManager.paintDirtyRegions(RepaintManager.java:769)
    at javax.swing.RepaintManager.prePaintDirtyRegions(RepaintManager.java:718)
    at javax.swing.RepaintManager.access$1100(RepaintManager.java:62)
    at javax.swing.RepaintManager$ProcessingRunnable.run(RepaintManager.java:1677)
    at java.awt.event.InvocationEvent.dispatch(InvocationEvent.java:251)
    at java.awt.EventQueue.dispatchEventImpl(EventQueue.java:733)
    at java.awt.EventQueue.access$200(EventQueue.java:103)
    at java.awt.EventQueue$3.run(EventQueue.java:694)
    at java.awt.EventQueue$3.run(EventQueue.java:692)
    at java.security.AccessController.doPrivileged(Native Method)
    at java.security.ProtectionDomain$1.doIntersectionPrivilege(ProtectionDomain.java:76)
    at java.awt.EventQueue.dispatchEvent(EventQueue.java:703)
    at java.awt.EventDispatchThread.pumpOneEventForFilters(EventDispatchThread.java:242)
    at java.awt.EventDispatchThread.pumpEventsForFilter(EventDispatchThread.java:161)
    at java.awt.EventDispatchThread.pumpEventsForHierarchy(EventDispatchThread.java:150)
    at java.awt.EventDispatchThread.pumpEvents(EventDispatchThread.java:146)
    at java.awt.EventDispatchThread.pumpEvents(EventDispatchThread.java:138)
    at java.awt.EventDispatchThread.run(EventDispatchThread.java:91)

最佳答案

您可能在 EDT 之外对 GUI 执行操作。在您怀疑的每一段代码中,您可能想要使用 <a href="http://docs.oracle.com/javase/7/docs/api/javax/swing/SwingUtilities.html#isEventDispatchThread%28%29" rel="noreferrer noopener nofollow">SwingUtilities.isEventDispatchThread()</a>测试您当前是否处于 EDT。如果此方法返回 false那么您不应执行任何 UI 操作。

这是一个如何使用它进行调试的示例:

public static void checkInEventDispatchThread(String pMethod) {
    if (!SwingUtilities.isEventDispatchThread()) {
        System.err.println("** NOT in EDT: " + pMethod + "() in " + 
                           Thread.currentThread().getName());
    }
}

对于您怀疑的每一段代码:

public void myMethod() {
    UtilClass.checkInEventDispatchThread("myMethod");
}

当然,这只是为了调试目的,你不能保留这样的调用用于生产/发布。

关于Java Swing - 如何判断什么代码在组件上调用 paint?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/21356691/

相关文章:

c++ - 原子操作 C++

java - 比一系列强制转换更好的关闭子窗口的方法?

java - 将字符串添加到字符串数组的开头

java - 如何防止在java中的数组列表中存储特殊字符/*和*/之间的文本

java - 点击ImageView后如何启动drawable动画?

java - 我可以使用 mouselistener/mouseadapter 访问 label[][] 中的所有标签吗?

对象之间的Java Swing间距

java - java中基于参数的HTTP Get方法

java - 内存分配给从 servlet 执行的主类?

ios - 非 ui 更新功能,例如读取 get 属性,是否需要在主线程中执行