打开JFileChooser时发生IOException

7

好的,这个问题真的很奇怪。每次我的应用程序打开JFileChooser时,它都会抛出一个IOException异常,然后一些图标显示不正确。

java.io.IOException
    at sun.awt.image.GifImageDecoder.readHeader(GifImageDecoder.java:265)
    at sun.awt.image.GifImageDecoder.produceImage(GifImageDecoder.java:102)
    at sun.awt.image.InputStreamImageSource.doFetch(InputStreamImageSource.java:246)
    at sun.awt.image.ImageFetcher.fetchloop(ImageFetcher.java:172)
    at sun.awt.image.ImageFetcher.run(ImageFetcher.java:136)

现在,当我深入分析这个错误时,似乎只有一个图标在尝试读取头文件时仅检索到了前8个字节,这是不够的。我已经检查了所有图标文件,它们都是正常的。 我尝试用另一个在此错误之前正确加载的图标文件来覆盖该图标文件,但结果相同。
以下是我的堆栈跟踪信息:
Daemon Thread [Image Fetcher 0] (Suspended (exception IOException)) 
    GifImageDecoder.readHeader() line: 265 [local variables unavailable]    
    GifImageDecoder.produceImage() line: 102 [local variables unavailable]  
    ByteArrayImageSource(InputStreamImageSource).doFetch() line: 246    
    ImageFetcher.fetchloop() line: 172  
    ImageFetcher.run() line: 136 [local variables unavailable]

当我深入GifImageDecoder实例时,这是我的变量值。

source  ByteArrayImageSource  (id=272)  
awaitingFetch   false   
consumers   null    
decoder GifImageDecoder  (id=271)   
decoders    GifImageDecoder  (id=271)   
imagedata    (id=307)   
    [0] 71  
    [1] 73  
    [2] 70  
    [3] 56  
    [4] 57  
    [5] 97  
    [6] 16  
    [7] 13  
    [8] 10  
imagelength 9   
imageoffset 0   

通常,这个图像数据应该更大。前10个字节是头文件,但是只检索了8个字节,如您所见。在此异常之后,JFileChooser中的每个其他图标都无法正确加载。

这是一个正确调用readHeader()的示例:

source  ByteArrayImageSource  (id=208)  
awaitingFetch   false   
consumers   null    
decoder GifImageDecoder  (id=207)   
decoders    GifImageDecoder  (id=207)   
imagedata    (id=223)   
    [0...99]    
    [100...199] 
    [200...299] 
    [300...399] 
    [400...499] 
    [500...599] 
    [600...699] 
    [700...799] 
    [800...899] 
    [900...979] 
imagelength 980 
imageoffset 0   

缓冲区已经完全加载了一个图标,就在抛出错误的那个图标之前。
以下是可能导致程序崩溃的示例(这种情况在我的代码中的多个部分都发生过,每当我首次加载系统图标时):
public class DirectoryBrowser extends JFileChooser{

private String suffixAccepted = null;

public DirectoryBrowser(File file, String chooserTitle, String approveOpenBtnText, String suffixAccepted)
{
    super(file);
    this.suffixAccepted = suffixAccepted;
    init(chooserTitle, approveOpenBtnText);
}

当它进入超级(file)时,它会到那里:
Thread [AWT-EventQueue-0] (Suspended)   
    Object.wait(long) line: not available [native method]   
    MediaTracker.waitForID(int, long) line: 651 
    ImageIcon.loadImage(Image) line: 234    
    ImageIcon.<init>(byte[]) line: 215  
    SwingUtilities2$2.createValue(UIDefaults) line: 1105    
    UIDefaults.getFromHashtable(Object) line: 185   
    UIDefaults.get(Object) line: 130    
    MultiUIDefaults.get(Object) line: 44    
    MultiUIDefaults(UIDefaults).getIcon(Object) line: 411   
    UIManager.getIcon(Object) line: 613 
    IronFileChooserUI(BasicFileChooserUI).installIcons(JFileChooser) line: 233  
    IronFileChooserUI(BasicFileChooserUI).installDefaults(JFileChooser) line: 219   
    IronFileChooserUI(BasicFileChooserUI).installUI(JComponent) line: 135   
    IronFileChooserUI(MetalFileChooserUI).installUI(JComponent) line: 139   
    DirectoryBrowser(JComponent).setUI(ComponentUI) line: 653   
    DirectoryBrowser(JFileChooser).updateUI() line: 1757    
    DirectoryBrowser(JFileChooser).setup(FileSystemView) line: 366  
    DirectoryBrowser(JFileChooser).<init>(File, FileSystemView) line: 332   
    DirectoryBrowser(JFileChooser).<init>(File) line: 315   
    DirectoryBrowser.<init>(File, String, String, String) line: 33  
    PackToIntegratePanel.choosePackPathToIntegrateFile() line: 522  
    PackToIntegratePanel$1.actionPerformed(ActionEvent) line: 104   
    JButton(AbstractButton).fireActionPerformed(ActionEvent) line: 1849 
    AbstractButton$Handler.actionPerformed(ActionEvent) line: 2169  
    DefaultButtonModel.fireActionPerformed(ActionEvent) line: 420   
    DefaultButtonModel.setPressed(boolean) line: 258    
    BasicButtonListener.mouseReleased(MouseEvent) line: 236 
    JButton(Component).processMouseEvent(MouseEvent) line: 5517 
    JButton(JComponent).processMouseEvent(MouseEvent) line: 3135    
    JButton(Component).processEvent(AWTEvent) line: 5282    
    JButton(Container).processEvent(AWTEvent) line: 1966    
    JButton(Component).dispatchEventImpl(AWTEvent) line: 3984   
    JButton(Container).dispatchEventImpl(AWTEvent) line: 2024   
    JButton(Component).dispatchEvent(AWTEvent) line: 3819   
    LightweightDispatcher.retargetMouseEvent(Component, int, MouseEvent) line: 4212 
    LightweightDispatcher.processMouseEvent(MouseEvent) line: 3892  
    LightweightDispatcher.dispatchEvent(AWTEvent) line: 3822    
    WorkbenchFrame(Container).dispatchEventImpl(AWTEvent) line: 2010    
    WorkbenchFrame(Window).dispatchEventImpl(AWTEvent) line: 1791   
    WorkbenchFrame(Component).dispatchEvent(AWTEvent) line: 3819    
    EventQueue.dispatchEvent(AWTEvent) line: 463    
    EventDispatchThread.pumpOneEventForHierarchy(int, Component) line: 242  
    EventDispatchThread.pumpEventsForHierarchy(int, Conditional, Component) line: 163   
    EventDispatchThread.pumpEvents(int, Conditional) line: 157  
    EventDispatchThread.pumpEvents(Conditional) line: 149   
    EventDispatchThread.run() line: 110 

然后上面提到的另一个线程检索图标(参见本帖子的第二个堆栈)。

你在处理事件队列吗? - Catalina Island
你在处理事件队列吗?什么意思?我没有操作任何队列。 - TecHunter
图标存储在哪里?是“默认”JFileChooser图标还是你们的图标?请添加一些代码,这会有所帮助。 - Charliemops
1
Catalina Island 意味着事件分派线程。如果代码在 EDT 中执行。 - Kai Mechel
我使用Java的默认lookAndFeel,所以我的图标在jdk libs中。而且显然它运行在EDT上,我会尝试将其移动到另一个独占线程。或者也许在应用程序的第一次启动时,我可以加载外观,这样它就可以在单独的线程上加载图像,然后就不会在EDT上加载时崩溃了。有朝一日会尝试,谢谢。 - TecHunter
显示剩余2条评论
2个回答

1
你的意思是什么?我没有操作任何队列。
抱歉,就像@kai所说,我指的是事件分派线程(EDT)。确保你创建所有组件,包括JFileChooser,都在EDT上,就像教程中所示。如果不这样做,它将成为某个文件系统和你的显示之间的竞争。
我刚注意到:没有那个IronFileChooser的情况下还会发生吗?

我尝试在进入操作时使用invokeLater,但仍存在错误。但是我认为你对线程的想法是正确的,我需要重新设计整个程序,并确保它在EDT中运行。 - TecHunter
IronFileChooser只是外观,我已经尝试过没有它,但结果仍然一样。无论如何,你是最接近答案的人,现在我只需要修复它。谢谢。 - TecHunter
1
好的,这值得一试。你可以尝试使用 ThreadCheckingRepaintManager 来查找EDT问题。 - Catalina Island

0

我曾经遇到过非常类似的问题。似乎从1.6.X版本开始,这是JFileChooser的一个bug。有时候它会无缘无故地抛出这个内部异常。我不确定它是否已经被修复或者是否有适当的文档记录。此外,我也不确定这是否适用于你的情况,但它确实让我想起了这个问题。


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