0

我需要移动/重命名一些加载在图片框内的图像,这些图像存储在应用程序可执行路径中的文件夹中。在我可以移动/重命名任何图像之前,我必须确保它没有被使用,所以我尝试将另一个图像从应用程序的可执行路径加载到这个图片框,并尝试将图片框图像属性“null”。

如果我想在我的程序中继续前进,图像仍然被锁定,我使用以下代码验证了这一点:

protected virtual bool IsFileLocked(FileInfo file)
    {
        FileStream stream = null;

        try
        {
            stream = file.Open(FileMode.Open, FileAccess.Read, FileShare.None);
        }
        catch (IOException)
        {
            return true;
        }
        finally
        {
            if (stream != null)
                stream.Close();
        }
        return false;
    }

我还尝试使用此处的代码:什么进程锁定了文件?

这给了我以下信息:System.Collections.Generic.List´1[System.Diagnostics.Process]

我必须告诉你,我也在使用相同的图像和方法与 AccessViolationException 作斗争。我不完全确定这是否是同一个问题......是我保存图像的方式错误还是我使用它们的方式?如果您需要更多代码或信息,请告诉我!

编辑:

来自 TaW 的链接和来自 Anon Coward 的答案都对我不起作用,我真的尽我最大的努力让它工作,我试图处理并取消图片框、图像等。我找到了一个与你的建议混合的解决方案,但是这是一个非常肮脏的...

var image2 = Image.FromFile(infos[i].FullName);
                                MemoryStream ms = new MemoryStream();
                                if (ImageFormat.Jpeg.Equals(image2.RawFormat))
                                {
                                    // JPEG
                                    image2.Save(ms, ImageFormat.Jpeg);
                                }
                                else if (ImageFormat.Png.Equals(image2.RawFormat))
                                {
                                    // PNG
                                    image2.Save(ms, ImageFormat.Png);
                                }
                                else if (ImageFormat.Gif.Equals(image2.RawFormat))
                                {
                                    // Bitmap
                                    image2.Save(ms, ImageFormat.Bmp);
                                }
                                else if (ImageFormat.Icon.Equals(image2.RawFormat))
                                {
                                    // Icon
                                    image2.Save(ms, ImageFormat.Icon);
                                }
                                image2.Dispose();

                                System.GC.Collect();
                                System.GC.WaitForPendingFinalizers();

                                System.IO.File.Delete(infos[i].FullName);

                                Image newImg = Image.FromStream(ms);
                                newImg.Save(neuer_name2);
                                newImg = null;

我知道使用 GarbageCollection 不是正确的方法,但我不知道是什么阻止了图像......所以我仍然在这里寻找真正的解决方案。

4

1 回答 1

2

当您将该属性设置为 null 时,您只是将对象标记为在将来某个时间点可用于垃圾收集,因此该对象的文件句柄将保持打开状态直到将来某个时间点。

如果您打算修改文件,您需要做两件事:将属性设为 Null 以使 PictureBox 不会尝试使用它,以及释放 Image 本身以关闭文件句柄:

var image = pictureBox1.Image;
pictureBox1.Image = null;
image.Dispose();

此外,您可能会遇到 GDI+ 的问题。您首先需要防止 Image 构造函数保留对文件的引用,您可以通过先将其读入 MemoryStream 轻松做到这一点,然后将其传递给构造函数:

Image image = Image.FromStream(new MemoryStream(File.ReadAllBytes("test.png")))
于 2016-07-09T19:08:24.327 回答