0

我正在用 C# 编写一个 Windows 窗体应用程序,我需要在客户端 PC 上安装的相关程序中向用户打开一个文档/文件(.doc、.pdf、.xlsx、.tiff 等)。

该文件应在用户关闭显示程序后立即删除。

我尝试了几种创建和打开文件的选项,但还没有找到金蛋。

public static void databaseFileRead(string file_name, byte[] file)
{
    path = file_name;
    int file_size_int = file.Length;
    FileStream create = new FileStream(@path, FileMode.Create, FileAccess.ReadWrite, FileShare.Read, file_size_int, FileOptions.DeleteOnClose);
    create.Write(file, 0, file_size_int);
    FileStream open = File.Open(@path, FileMode.Open, FileAccess.ReadWrite, FileShare.Read);
}

在上述方法中,我得到一个 IOException,在最后一行 ( )上显示“进程无法访问文件 'xxx',因为它正被另一个进程使用”。FileStream open = ...

public static void databaseFileRead(string file_name, byte[] file)
{
    path = file_name;
    int file_size_int = file.Length;
    FileStream create = File.OpenWrite(@path);
    var attributes = File.GetAttributes(@path);
    File.SetAttributes(@path, attributes | FileAttributes.ReadOnly|FileAttributes.Temporary);
    create.Close();
    Process p = Process.Start(@path);
    p.WaitForExit();
    File.Delete(@path);
}

在这种方法中,我还收到一个 IOException 说明“该进程无法访问文件'xxx',因为它正在被另一个进程使用”在最后一行(File.Delete(@path);)意味着该文件仍在使用中,这是正确的。似乎p.WaitForExit();不是在等待所有程序,例如 OpenOffice ......

是否可以打开/显示FileOptions.DeleteOnClose在外部程序中创建的文件?如果是这样,怎么做?

我确实喜欢这个想法,Windows 会在文件不再使用时立即删除该文件。

重要的是文件会自动从用户硬盘中消失,最好的选择是从流或同等文件中读取和打开文件。但据我所知,这是不可能的......

解决了:

但是我不确定这是否是一种捕捉异常并closeIfReady再次调用该方法直到文件被释放的探测方法......

   public static void databaseFileRead(string file_name, byte[] file)
    {
        var path = file_name;
        int file_size_int = file.Length;
        if (File.Exists(path))
        {
            File.Delete(path);
        }
        FileStream create = File.OpenWrite(path);
        create.Write(file, 0, file_size_int);
        var attributes = File.GetAttributes(path);
        File.SetAttributes(path, attributes | FileAttributes.Temporary);
        create.Close();
        Process p = Process.Start(path);

        while (!p.HasExited)
        {
            Thread.Sleep(500);
        }
        closeIfReady(path);
    }

    static void closeIfReady(string path)
    {
        try
        { File.Delete(@path); }
        catch
        {
            Thread.Sleep(1000);
            closeIfReady(path);
        }
    }
4

2 回答 2

3

好的,在评论之后,这是一个有效的第二种方法:

void Method2(string file_name, byte[] file)
{
    var path = file_name;
    int file_size_int = file.Length;
    if (File.Exists(path))
    {
        File.Delete(path);
    }
    FileStream create = File.OpenWrite(path);
    var attributes = File.GetAttributes(path);
    File.SetAttributes(path, attributes | FileAttributes.Temporary);
    create.Close();
    Process p = Process.Start(path);

    while (!p.HasExited)  
    {
        Thread.Sleep(100);  
    }

    File.Delete(path);
}

您一开始并不严格需要 if File.Exists/File.Delete,我只需要它来调试我的 - 但是将其保留在那里可能是个好主意,因为如果有人在文件被删除之前退出应用程序它将尝试再次创建它。

然而,这种方法的问题是: File.SetAttributes(@path, attributes | FileAttributes.ReadOnly|FileAttributes.Temporary);因为文件是作为只读文件创建的,所以不能使用 File.Delete 删除它。在 Windows 资源管理器中,如果您有管理员,它不关心这一点,这就是为什么您可以从那里删除它,但 C# 确实关心文件属性。我将该行更改为File.SetAttributes(@path, attributes | FileAttributes.Temporary);,现在可以使用。

这个问题是我用于您的 p.WaitForExit 问题的参考。我建议在您的应用程序中的不同线程上启动它,否则您的应用程序将在检查线程退出时挂起(在 Windows 中变为白色/灰色并告诉您它没有响应)。如果您正在使用 wpf,您可以做一些事情,例如使视觉效果无效,或者重新绘制您的 GUI,以便您仍然可以使用它,但这些都是 hacky 并且会消耗大量 CPU。

于 2013-03-20T00:45:27.140 回答
0

有几件事。

首先,如果这是在 winforms 应用程序的主 UI 线程中运行的,那么使用任一解决方案,您的应用程序都将变得无响应。

其次,这两种解决方案都依赖于从几个不同的应用程序中优雅地退出。这是有风险的。

如果外部进程未能正确清理文件句柄,您的解决方案将导致无限递归。取决于 Process.HasExited,我也会感到不舒服,因为我已经看到许多关于此属性可靠性的线程。

我建议使用相对于您的应用程序的临时目录,并在应用程序退出和启动时清理该目录(以防您自己的应用程序无法正常退出)。

于 2013-03-27T21:36:48.897 回答