Directory.EnumerateFiles和File.Exists忽略本地DLL文件

4

环境:C#,.NET 4.0和混合模式。

我有一个复杂的应用程序,涉及多个第三方依赖项。当我们接到一些应用程序崩溃的报告时,我们添加了一些健全性检查以确保遍历程序集依赖项时所有依赖项都存在。不幸的是,这并不能检查我们的应用程序使用的本机DLL文件的存在。我们打算的解决方案是遍历所有DLL名称,并确保至少有一个文件具有该名称作为健全性检查。

问题

Directory.EnumerateFiles()和File.Exists()也看不到这些本机DLL。重现此问题的代码与教科书中的“如何列出文件”一样简单:

foreach(string file in
        Directory.EnumerateFiles(Environment.CurrentDirectory, "*.dll"))
{
    string entry = Path.GetFileName(file);
    if (! RequiredFiles.Contains(entry))
    {
        /* Do error handling */
    }
}

从我列出文件的目录开始,我可以看到我想要检测的文件。它们在任何方面都没有被标记为系统文件。然而,无论我是否有过滤文本,只有.NET DLL文件被列出。我想更直接地重写代码部分,但令人沮丧的是得到了相同的结果:

foreach(string dependency in RequiredFiles)
{
    string fileName = Environment.CurrentDirectory + '\\' + dependency;
    if(! File.Exists(fileName))
    { /* do error handling */ }
}

我得到了完全相同的结果。所有本地DLL文件似乎对.NET不可见。
问题:
这是什么原因?更重要的是,如果我甚至无法在文件系统中看到它们,如何检测本地DLL文件是否存在?

你能在一个空白项目中重现这个问题吗?我非常怀疑本地DLL会像那样“看不见”。听起来好像还有其他的事情发生了;当前目录错误,某种类型的阴影复制,64位操作系统上的32位进程文件系统重定向等等。ClickOnce部署可能会忽略本地DLL。诸如此类的东西。 - Roman Starkov
3个回答

2

您确定工作目录正确吗?否则Environment.CurrentDirectory将不会指向您期望的位置。如果DLL文件与您的执行代码位于同一目录中,则可以执行以下操作:

Assembly.GetExecutingAssembly().Location

检查了路径,它和我正在查看的是同一个。 - Berin Loritsch

2
  • 为你自己的DLL起一个名字,例如MySuperNiftyLibrary.dll。它可以只是一个重命名的文本文件。将其放在你正在查看的文件夹中。再次运行代码。看看那个名称是否列出。如果这不能说服你,可以创建一个小的.NET类库项目,使用一个独特的名称,并将其放在该目录中。

    没有理由Directory.EnumerateFiles会过滤掉.NET DLL而不是其他类型的DLL,因此你可能没有查看正确的目录。

  • 但如果你是,那么考虑一下你要查找的DLL是在顶级目录还是子目录中。默认情况下,EnumerateFiles仅列出顶级目录中的文件。要列出所有子目录中的所有文件,需要使用EnumerateFiles的此重载:

Directory.EnumerateFiles(directory, "*.dll", SearchOption.AllDirectories)


现在我感觉自己像个白痴。经过反复检查,我确信自己正在查看正确的目录。直到你发布了这个答案之后,我才发现我实际上是在查看一个非常相似但不同的目录。唯一缺少的文件就是本地DLL文件... - Berin Loritsch

0

然后使用shell,类似这样的代码应该可以工作:

            Shell32.Shell shl = null;
            Shell32.Folder folder = null;
            try
            {
                shl = new Shell32.Shell();
                folder = shl.NameSpace(Environment.CurrentDirectory);
                foreach (Shell32.FolderItem file in folder.Items())
                {
                    if (!RequiredFiles.Contains(file.path))
                    {/* do error handling */}
                }
            }
            catch
            { }
            finally
            {
                if (folder != null)
                {
                    System.Runtime.InteropServices.Marshal.FinalReleaseComObject(folder);
                    folder = null;
                }
                if (shl != null)
                {
                    System.Runtime.InteropServices.Marshal.FinalReleaseComObject(shl);
                    shl = null;
                }
            }

更新:如果您打算在XP和更新的操作系统上使用它,您需要在XP上引用Shell32.dll(Microsoft Shell控件和自动化)并编译一次,以便将此类dll复制到您的项目中。因为来自Vista或7的dll在XP上不起作用,反之亦然。这是一种更简单的方法,依赖于Shell脚本。否则,您可以直接与Shell API进行交互,但这更加复杂...


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