1

我昨天做了这个问题,但目前我没有得到任何答案。

无论如何,我的新方法是创建一个小程序在后台一直运行,并定期检查是否有临时文件未被应用程序使用。

这次我将在系统临时文件夹中创建一个文件夹来存储打开的文件。

这是代码:

private const uint GENERIC_WRITE = 0x40000000;
private const uint OPEN_EXISTING = 3;

private static void Main()
{
    while (true)
    {
        CleanFiles(Path.GetTempPath() + "MyTempFolder//");
        Thread.Sleep(10000);
    }
}

[DllImport("kernel32.dll", SetLastError = true, CharSet = CharSet.Unicode)]
private static extern SafeFileHandle CreateFile(string lpFileName, UInt32 dwDesiredAccess, UInt32 dwShareMode,
                                                IntPtr pSecurityAttributes, UInt32 dwCreationDisposition,
                                                UInt32 dwFlagsAndAttributes, IntPtr hTemplateFile);

private static void CleanFiles(string folder)
{
    if (Directory.Exists(folder))
    {
        var directory = new DirectoryInfo(folder);

        try
        {
            foreach (var file in directory.GetFiles())
                if (!IsFileInUse(file.FullName))
                {
                    Thread.Sleep(5000);
                    file.Delete();
                }
        }
        catch (IOException)
        {
        }
    }
}


private static bool IsFileInUse(string filePath)
{
    if (!File.Exists(filePath))
        return false;

    SafeHandle handleValue = null;

    try
    {
        handleValue = CreateFile(filePath, GENERIC_WRITE, 0, IntPtr.Zero, OPEN_EXISTING, 0, IntPtr.Zero);
        return handleValue.IsInvalid;
    }
    finally
    {
        if (handleValue != null)
        {
            handleValue.Close();
            handleValue.Dispose();
        }
    }
}

但这有一个问题:

它适用于 docx 和 pdf(使用 Foxit Reader)文件。

即使记事本仍在使用 txt 文件,它们也会被删除,但我可以忍受这一点,因为文件的内容在记事本中仍然可见。

真正的问题在于 Windows Photo Viewer 等应用程序。即使图像仍在被 WPV 使用,图像也会被删除,但这次图像会从 WPV 中消失,并在其屏幕上显示消息正在加载...

我需要一种方法来真正检测应用程序是否仍在使用文件。

4

2 回答 2

1

你就是不能。

“文件被另一个程序使用”没有黑魔法。这仅仅意味着另一个程序已经打开了该文件的句柄。

一些应用程序会一直保持句柄打开,而其他应用程序(例如记事本)则不会:当您打开文件时,记事本会打开文件的句柄,通过打开的句柄读取整个文件,关闭句柄并显示向用户读取字节。

如果你删除文件,很好,没有打开的句柄,记事本不会注意到你删除了文件。

于 2013-06-24T11:57:10.973 回答
0

Please take a look into this SO question

Here you can check the application by the application name(easier way):

 Process[] pname = Process.GetProcessesByName("notepad");
 if (pname.Length == 0)
    MessageBox.Show("nothing");
 else
    MessageBox.Show("run");
于 2013-06-24T11:54:52.157 回答