2

我有一个class GridPanel extends JPanel, 带有一个静态内部类ToolSelectComboBox extends JComboBox,它又具有两个静态内部类ToolSelectComboBoxModel implements ComboBoxModelToolSelectComboBoxRenderer implements ListCellRenderer. 面板显示ToolSelectComboBox(TSCB),其构造函数将其模型和渲染器设置为我创建的自定义渲染器。盒子创建正确,其模型和渲染器工作正常。

但是,渲染器的getListCellRendererComponent(...)方法在它返回ImageIcon的那个上使用了一个。JLabel该图标已正确加载,但是,当我第一次单击组合框(每次运行时)时,图像的加载时间恰好(或至少非常接近)一秒多一点。我会假设这是加载文件的一些滞后,除了

  • 这是我本地文件系统上的一个 4kB 文件
  • 当我在System.out.println命令之前和之后添加命令时result.setIcon(...),它们几乎立即相互跟随。

我注意到的奇怪的事情是println命令被触发了两次,一次是在我单击框时,一次是在图标加载时。

还可能值得注意的是,由于它旨在与覆盖父抽象类的单个方法的多个类一起使用(以生成图标的路径),当我注意到它工作缓慢时,我将代码从只需使用getIcon命令检索图标以将各种大小的图标(16、32 和 64 px 平方)存储在一个TreeMap<Tool.ImageSize, ImageIcon>Tool我创建的具有ImageIcon getIcon()方法的界面在哪里。

我所有的进口都井井有条。

任何帮助,将不胜感激!

如果我发布了太多代码,我深表歉意,但我想确保它是可以理解的。另一方面,如果您需要更多代码才能理解,请不要犹豫。

代码(所有以“”开头*并具有类似注释的文本的行都是折叠的 JavaDoc 标记,而不仅仅是混乱的代码):

public class GridPanel extends JPanel {

    public static class ToolSelectComboBox extends JComboBox {

         // Combo box model `ToolSelectComboBoxModel` snipped

         * A renderer for the {@link ToolSelectComboBoxModel}. This may
        public static class ToolSelectComboBoxRenderer implements
                ListCellRenderer {

             * The default renderer. Only the icon and text are modified.
            protected DefaultListCellRenderer d = new DefaultListCellRenderer();

            @Override
            public Component getListCellRendererComponent(final JList list,
                    final Object value, final int index,
                    final boolean isSelected, final boolean cellHasFocus) {
                if (!ToolSelectComboBoxModel.class.isInstance(list.getModel())) {
                    throw new IllegalStateException(
                            "Cannot use a ToolSelectComboBoxRenderer on any list model type other than ToolSelectComboBoxModel.");
                }
                final JLabel result = (JLabel) d.getListCellRendererComponent(
                        list, value, index, isSelected, cellHasFocus);
                result.setText(null);
                if (value != null) {
                    result.setIcon(((Tool) value)
                            .getIcon(Tool.IconSize.SIZE_32PX));
                }
                return result;
            }
        }

        public ToolSelectComboBox() {
            setModel(new ToolSelectComboBoxModel());
            ((ToolSelectComboBoxModel) getModel()).add(new CircleTool()); // shown below
            setRenderer(new ToolSelectComboBoxRenderer());
        }
    }

     * Create the panel.
    public GridPanel() {
        setLayout(new BorderLayout(0, 0));

        final ToolSelectComboBox toolSelectComboBox = new ToolSelectComboBox();
        add(toolSelectComboBox, BorderLayout.NORTH);

        final SquareGrid squareGrid = new SquareGrid(); // another class; this works fine
        add(squareGrid, BorderLayout.CENTER); // irrelevant to problem

    }

}

该类CircleTool只有一个方法(覆盖AbstractTool的抽象方法以获取图像路径),并且由于该方法有效(它可以很好地获取路径,它只是加载缓慢的图标),因此我没有包含此类。

AbstractTool班级:

public abstract class AbstractTool implements Tool {

    /**
     * A {@link TreeMap} to map the icon sizes to their icons.
     */
    protected final TreeMap<Tool.IconSize, ImageIcon> map = new TreeMap<Tool.IconSize, ImageIcon>();

    /**
     * Constructs the tool and sets up the {@linkplain #map}.
     */
    public AbstractTool() {
        for (final Tool.IconSize size : Tool.IconSize.values()) {
            System.out.println("Putting value for " + size);
            map.put(size,
                    new ImageIcon(Tool.class.getResource(getImagePath(size))));
        }
    }

    @Override
    public ImageIcon getIcon(final IconSize size) {
        return map.get(size);
    }

    /**
     * Gets the image path for the given image size.
     * 
     * @param size
     *            the size
     * @return the image path
     */
    protected abstract String getImagePath(Tool.IconSize size);

}
4

1 回答 1

3

但是,我第一次单击组合框(每次运行)时,加载图像需要一秒钟多一点的时间。我认为这是加载文件的一些滞后

这也是我的猜测。

除了当我在 result.setIcon(...) 命令之前和之后添加 System.out.println 命令时,它们几乎立即相互跟随

当您单击组合框时,所有代码都会在 EDT 上运行,这意味着每个图标将按顺序加载。

但是 System.out.println() 在单独的线程上运行,因此会立即显示。

解决方案是在程序启动时加载图标。也就是说,每当您在地图中定义/添加图标时,您都应该在那时阅读它们。您可能希望在单独的线程上执行此操作,以免阻止 GUI 显示。

编辑:

这是一个简单的 SSCCE,它在组合框中显示图标:

import java.awt.*;
import javax.swing.*;

public class ComboBoxIcon extends JFrame
{
    JComboBox comboBox;

    public ComboBoxIcon()
    {
        Object[] items =
        {
            new ImageIcon("about16.gif"),
            new ImageIcon("add16.gif"),
            new ImageIcon("copy16.gif")
        };
        comboBox = new JComboBox( items );
        getContentPane().add( comboBox, BorderLayout.NORTH );
    }

    public static void main(String[] args)
    {
        JFrame frame = new ComboBoxIcon();
        frame.setDefaultCloseOperation( EXIT_ON_CLOSE );
        frame.pack();
        frame.setLocationRelativeTo( null );
        frame.setVisible( true );
    }
}

如果您需要更多帮助,则需要发布您的SSCCE来证明问题。

于 2011-09-11T18:53:08.870 回答