JFileChooser 在使用窗口外观时非常慢



当使用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);
    }
}

我遇到了同样的问题,至少就我而言,它与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 
    ...

所以我看到两个解决方案:

正如Joop Eggen所提到的,在你需要它之前很久就初始化你的文件选择器,或者安装你自己的UI,它会尝试在额外的线程中检索图标,只返回那些立即可用的图标。

一个解决方案(很多年前(是使用 java.awt FileChooser。更好的似乎是在后台预加载一个共享的JFileChooser:

一个有未来的领域:

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

然后在初始化时执行以下操作:

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

因此,当最终需要JFileChooser时,请从未来获取它。

JFileChooser fileChooser = futureFileChooser.get();

如果在加载时间内完成此操作,它仍然会阻塞,但已经停留了更少的时间。当做的第一件事是打开 JFileChooser 时,此解决方案将不起作用。

我可能无法说服你切换到JavaFX,如果本机的外观和感觉是如此需要的话。

这段代码帮助了我,从那时起我没有发现任何问题。

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

对我来说,解决方案是从桌面上删除LibreOffice Writer和Calc的快捷方式,这些快捷方式是在安装过程中自动创建的。手动制作了另一个快捷方式(右键单击拖放(,这些快捷方式并没有减慢文件选择器的打开速度。

只是添加到知识库中。在 win 10 java 18 上

javax.swing.UIManager$LookAndFeelInfo[CDE/Motif com.sun.java.swing.plaf.motif.MotifLookAndFeel]

工作速度非常快。注意:在上面的主题和com.之间有一个空格。

最新更新