使用Windows外观时,JFileChooser非常缓慢。

6

当我使用JFileChooser和Windows外观时,负载时间约为21-40秒。如果我删除外观,代码运行速度非常快。我猜问题出在初始化new JFileChooser("path");时。有人能帮我吗?以下是我的代码:

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

public class MainClass {

    public static void main(String[] args) {
        EventQueue.invokeLater(new Runnable() {

            public void run() {
                createGUI();
            }
        });
    }

    private static void createGUI() throws HeadlessException {
        final JFrame frame = new JFrame("JFileChooser Demo");

        try
        {
            UIManager.setLookAndFeel(UIManager.getSystemLookAndFeelClassName());
        }
        catch (Exception e)
        {
            e.printStackTrace();
        }

        final JFileChooser fc = new JFileChooser();
        fc.setMultiSelectionEnabled(true);
        fc.setCurrentDirectory(new File("./feedback"));
        JButton btn1 = new JButton("Show Dialog");
        btn1.addActionListener(new ActionListener() {

            public void actionPerformed(ActionEvent e) {
                fc.showDialog(frame, "Choose");
            }
        });

        Container pane = frame.getContentPane();
        pane.setLayout(new GridLayout(3, 1, 10, 10));
        pane.add(btn1);

        frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        frame.pack();
        frame.setVisible(true);
    }
}

2
在哪个版本的Java和Windows上?我已经在Windows 7上尝试过这个,使用的是Java 1.8.0_66,没有任何问题(这个工作站不是最新的……)。还在一个虚拟机上测试了XP和Java 1.6.0_23(这个真的很老旧!!!)。 - AxelH
我正在使用Windows 10和Java 1.7.0_17。升级能修复它吗? - ch.Joshi elijah
刚刚更新了我的评论,它可能不是Java,但在Java 6和8上运行良好。但可能是Windows 10的外观和感觉。我没有在办公室里... - AxelH
@AxelH 请尝试导航到 C:\Windows\System32。图标加载速度较慢。 - Johannes Kuhn
1
我正在谈论初始加载时间,我的意思是实际的jfilechooser弹出时间,而不是任何文件的加载时间。我正在使用转速为5400 rpm的硬盘。我相信运行时间会因为固态硬盘而有所不同,如果您正在使用固态硬盘,请尝试更多次加载实际图形用户界面,您会发现有点卡顿。抱歉,我昨天没有注意到您的评论。 - ch.Joshi elijah
显示剩余6条评论
5个回答

8

我遇到了同样的问题,但至少在我的情况下,这与Windows或Java版本无关。

创建JFileChooser时,Windows外观已经尝试通过请求所有可见项(包括网络共享的图标)的图标来计算当前目录组合框的大小。在我的情况下,我的一些网络共享不可用。

这是我在Java 8和Windows 10上的堆栈跟踪:

Thread [AWT-EventQueue-0]
    Unsafe.park(boolean, long) line: not available [native method]  
    LockSupport.park(Object) line: 175  
    FutureTask<V>.awaitDone(boolean, long) line: 429    
    FutureTask<V>.get() line: 191
    Win32ShellFolderManager2$ComInvoker.invoke(Callable<T>) line: 582   
    ShellFolder.invoke(Callable<T>, Class<E>) line: 518 
    ShellFolder.invoke(Callable<T>) line: 504   
    Win32ShellFolder2.getIcon(boolean) line: 968    
    WindowsFileSystemView(FileSystemView).getSystemIcon(File) line: 243 
    WindowsFileChooserUI$WindowsFileView.getIcon(File) line: 1300   
    JFileChooser.getIcon(File) line: 1609   
    WindowsFileChooserUI$DirectoryComboBoxRenderer.getListCellRendererComponent(JList, Object, int, boolean, boolean) line: 998 
    BasicListUI.updateLayoutState() line: 1361
    BasicListUI.maybeUpdateLayoutState() line: 1311
    BasicListUI$Handler.valueChanged(ListSelectionEvent) line: 2623 
    DefaultListSelectionModel.fireValueChanged(int, int, boolean) line: 184 
    DefaultListSelectionModel.fireValueChanged(int, int) line: 164  
    DefaultListSelectionModel.fireValueChanged() line: 211  
    DefaultListSelectionModel.changeSelection(int, int, int, int, boolean) line: 405    
    DefaultListSelectionModel.changeSelection(int, int, int, int) line: 415 
    DefaultListSelectionModel.setSelectionInterval(int, int) line: 459  
    BasicComboPopup$1(JList<E>).setSelectedIndex(int) line: 2216    
    BasicComboPopup.setListSelection(int) line: 1186    
    BasicComboPopup.access$300(BasicComboPopup, int) line: 65   
    BasicComboPopup$Handler.itemStateChanged(ItemEvent) line: 1015  
    WindowsFileChooserUI$2(JComboBox<E>).fireItemStateChanged(ItemEvent) line: 1223 
    WindowsFileChooserUI$2(JComboBox<E>).selectedItemChanged() line: 1280   
    WindowsFileChooserUI$2(JComboBox<E>).contentsChanged(ListDataEvent) line: 1330  
    WindowsFileChooserUI$DirectoryComboBoxModel(AbstractListModel<E>).fireContentsChanged(Object, int, int) line: 118   
    WindowsFileChooserUI$DirectoryComboBoxModel.setSelectedItem(Object) line: 1140  
    WindowsFileChooserUI$DirectoryComboBoxModel.addItem(File) line: 1111    
    WindowsFileChooserUI$DirectoryComboBoxModel.access$800(WindowsFileChooserUI$DirectoryComboBoxModel, File) line: 1041    
    WindowsFileChooserUI.doDirectoryChanged(PropertyChangeEvent) line: 730  
    WindowsFileChooserUI.access$1100(WindowsFileChooserUI, PropertyChangeEvent) line: 55    
    WindowsFileChooserUI$11.propertyChange(PropertyChangeEvent) line: 821   
    PropertyChangeSupport.fire(PropertyChangeListener[], PropertyChangeEvent) line: 335 
    PropertyChangeSupport.firePropertyChange(PropertyChangeEvent) line: 327 
    PropertyChangeSupport.firePropertyChange(String, Object, Object) line: 263  
    JFileChooser(Component).firePropertyChange(String, Object, Object) line: 8434   
    JFileChooser.setCurrentDirectory(File) line: 598    
    JFileChooser.<init>(File, FileSystemView) line: 344 
    JFileChooser.<init>(File) line: 326 
    FileChooser.<clinit>() line: 15 
    ...

我看到两个解决方案:
1.像Joop Eggen提到的那样,在你需要它之前很长时间就初始化您的文件选择器,或者安装自己的UI,尝试在额外线程中检索图标,并仅返回那些立即可用的图标。 2. 安装自己的UI,尝试在额外的线程中检索图标,并只返回那些可以立即使用的图标。

1
天啊,太感谢了。虽然我也把问题缩小到了JFileChooser开始遍历目录结构的那一点,但我从来没有想过一个无法访问的网络共享会导致这种不便。 - Johnson
回应说我也遇到了同样的问题 - 我不知道我的网络驱动器掉了,突然间我的JFileChooser变得非常缓慢。我以为这是JDK 17的问题,因为我最近更新了,但清除坏的驱动器就足以使它恢复正常。 - lucasvw

2

其中一种解决方案(许多年前)是使用java.awt FileChooser。更好的选择似乎是在后台预加载共享的JFileChooser:

一个带有Future的字段:

FutureTask<JFileChooser> futureFileChooser = new FutureTask<>(JFileChooser::new);

在初始化时进行以下操作:

ExecutorService executor = Executors.newSingleThreadExecutor();
executor.execute(futureFileChooser);

当你需要使用JFileChooser时,可以从Future中获取它。

JFileChooser fileChooser = futureFileChooser.get();

如果在加载时间内完成此操作,仍会阻塞,但时间更短。当第一件事是打开JFileChooser时,这种解决方案将无法使用。
如果非要使用本地外观和感觉,我可能无法说服您转换到JavaFX。

如果我可以放弃查找,运行时间会非常短。@Joop Eggen。 - ch.Joshi elijah
有些外观很不错。Nimbus 有点像 Macintosh,但我相信 Plastic 和其他一些第三方主题可能适合 Windows。 - Joop Eggen

0

对我来说,解决方案是从桌面上移除LibreOffice Writer和Calc的快捷方式,这些快捷方式在安装过程中自动创建。手动制作了另一个快捷方式(使用右键拖放),这些快捷方式没有拖慢FileChooser的打开速度。


0

这段代码帮助了我,自那以后我没有发现任何问题。

WindowsFileChooserUI wui = new WindowsFileChooserUI(filechosser);
wui.installUI(filechosser);

0

补充一下知识库。在win 10 java 18上,javax.swing.UIManager$LookAndFeelInfo[CDE/Motif com.sun.java.swing.plaf.motif.MotifLookAndFeel]非常快。注意:上述代码中Motif和com之间有一个空格。


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接