java - 如果选择第一个条目,Swing JList 速度会极度减慢 (Windows 10)

标签 java swing jlist defaultlistmodel openjdk-11

目标:
将 JList 用于任何目的。

问题:
如果选择第一个条目,然后清除模型并添加新内容,无论您多久执行一次,这都会极其缓慢。仅当您选择了另一个条目(该条目不是第​​一个条目)时,行为才会再次快速=正常。

解决方法:
每次清除模型之​​前,以编程方式添加两个任意条目(两个,因为列表可能完全为空),然后选择有效索引最高的条目。

问题:
你能重现这个错误吗?这是一个已知的错误吗?如果没有,有人可以发布它以便 OpenJDK 得到修复吗?

测试环境:
标准 Oracle Java 发行版下载 JDK8 202。
java版本“1.8.0_202”
Java(TM) SE 运行时环境(内部版本 1.8.0_202-b08)
Java HotSpot(TM) 64 位服务器 VM(内部版本 25.202-b08,混合模式)
Windows 10 家庭版,版本 1809(操作系统内部版本 17763.437)
英特尔(R) 酷睿(TM) i7-6700K CPU @ 4.00 GHz 4.01 GHz

也用OpenJDK 11
复制 openjdk版本“11.0.2”2019-01-15
OpenJDK 运行时环境 18.9(内部版本 11.0.2+9)
OpenJDK 64 位服务器 VM 18.9(内部版本 11.0.2+9,混合模式)

SSCCE

此代码很少,因此它甚至使用默认的 LookAndFeel。最初,我在使用L&F“Windows”时遇到了这个问题。

package jlistslowdownbugdemo;

import javax.swing.*;
import java.awt.*;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Random;




final public class Main {


    final private static boolean ENABLE_FIX = false;


    public static void main(final String[] args) {


        SwingUtilities.invokeLater(() -> {


            final DefaultListModel<String> valueListModel = new DefaultListModel<>();
            final JList<String> valueList = new JList<>(valueListModel);


            final DefaultListModel<String> keyListModel = new DefaultListModel<>();
            for (String key : Main.DATA.keySet()) {
                keyListModel.addElement(key);
            }
            final JList<String> keyList = new JList<>(keyListModel);


            keyList.addListSelectionListener(e -> {

                if (e.getValueIsAdjusting()) { // To prevent SUPER-slowdown when the bug kicks. Not an important measure, just for comfortable demoing.


                    // Workaround for the bug: Add two entries (two, because list may be entirely empty),
                    // and select the last available entry (in the worst case, this is the 2nd).
                    // Just so that one is selected that IS NOT THE FIRST.
                    // This problem AND fix occurred in Oracle JDK 8 and Open JDK 11.
                    if (ENABLE_FIX) {
                        valueListModel.addElement(null);
                        valueListModel.addElement(null); // Make sure you add something else instead of null if your GUI/JList setup requires it.
                        valueList.setSelectedIndex(valueListModel.size() - 1);
                    }


                    valueListModel.clear();

                    final String key = keyList.getSelectedValue();
                    if (key != null) {
                        for (String value : Main.DATA.get(key)) {
                            valueListModel.addElement(value);
                        }
                    }

                }
            });


            final JPanel contentPane = new JPanel(new GridLayout(1, 0, 0, 0));
            contentPane.add(new JScrollPane(keyList));
            contentPane.add(new JScrollPane(valueList));


            final JFrame window = new JFrame();
            window.setContentPane(contentPane);
            window.setDefaultCloseOperation(WindowConstants.EXIT_ON_CLOSE);
            window.setMinimumSize(new Dimension(1200, 1000));
            window.setLocationRelativeTo(null);
            window.setVisible(true);


            final String msg = "Click&drag in the left list. This rapidly changes the content of the right list.\n" +
                    "\n" +
                    "Now select any entry EXCEPT the first entry of the right list. Click&drag again in the left list,\n" +
                    "it still works just as rapidly. Now click THE FIRST entry of the right list.\n" +
                    "\n" +
                    "If you NOW click&drag again in the left list, you will experience BRUTAL slowdown.\n" +
                    "\n" +
                    "Once Swing has calmed down, click any entry EXCEPT the first entry of the right list.\n" +
                    "Click&drag again in the left list - the problem is gone.\n" +
                    "\n" +
                    "The built-in workaround (off by default) simulates this.";

            JOptionPane.showMessageDialog(window,
                                          msg,
                                          "How to reproduce the Swing bug:",
                                          JOptionPane.INFORMATION_MESSAGE);


        });

    }


    final private static Random RAND = new Random(0);


    final private static Map<String, List<String>> DATA = createDataMap();


    private static Map<String, List<String>> createDataMap() {

        final Map<String, List<String>> ret = new HashMap<>();

        for (int i = 0; i < 30; i++) {


            final int listSize = 20 + RAND.nextInt(5000);


            final String key = generateRandomString('A');
            final List<String> value = new ArrayList<>(listSize);
            ret.put(key, value);


            for (int ii = 0; ii < listSize; ii++) {
                value.add(generateRandomString('a'));
            }

        }

        return ret;
    }


    private static String generateRandomString(final char baseChar) {

        final StringBuilder sb = new StringBuilder();

        final int len = 4 + RAND.nextInt(17);

        for (int i = 0; i < len; i++) {
            sb.append((char) (baseChar + RAND.nextInt(26)));
        }

        return sb.toString();
    }


}

(标题已编辑:删除了“,绝对是一个 Java 错误”)

最佳答案

问题出在值列表的原型(prototype)单元格值上,因为现在我已将解决方案缩减为一行:设置此值:

valueList.setPrototypeCellValue("                ");

再说一遍,我不知道为什么会这样,但问题一定出在这里,因为这解决了一切。也许这与列表如何与 JScrollPane 交互有关,因为原型(prototype)单元格值是计算可滚动视口(viewport)大小和首选大小的关键。

关于java - 如果选择第一个条目,Swing JList 速度会极度减慢 (Windows 10),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/55936230/

相关文章:

java - longBitsToDouble,不进行舍入,是否可以

java - HashMap 冲突会导致调整大小吗?

java 鼠标右键双击事件与 Mac OS X 上的鼠标左键双击事件行为相同

java - 访问在方法中创建的 Swing 组件

java - 在运行时更改 FormLayout 对齐方式

java - 向 JList 添加项目导致 java.lang.NullPointerException

java - Swing 对 JList 的 MVC 实现有问题吗?

java - 面板任何元素的 KeyPressed 事件

java - 如何解决/bin/sh : protoc: command not found?

java - NetBeans 默认 Jlist 将其更改为 arraylist