Directory.EnumerateFiles和File.Exists正在忽略本机DLL文件
本文关键字:本机 DLL 文件 EnumerateFiles File Exists Directory | 更新日期: 2023-09-27 18:23:52
环境:C#、.NET 4.0和混合模式。
我有一个复杂的应用程序,有几个第三方依赖项。当我们收到一些应用程序崩溃的报告时,我们添加了一些健全性检查,通过遍历Assembly依赖项来确保所有依赖项都存在。不幸的是,这并不能检查我们的应用程序使用的本机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编一个名称,例如MySuperNiftyLibrary.DLL。它可以只是一个重命名的文本文件。把它放在你正在查看的文件夹中。再次运行代码。查看是否列出了该名称。如果这还不能说服你,那么创建一个具有独特名称的小型.NET类库项目,并将其放入目录中。
与其他类型的DLL相比,
Directory.EnumerateFiles
没有理由在.NET DLL上进行筛选,因此您不可能查看正确的目录。 -
但是,如果是,请考虑您要查找的DLL是在顶级目录中还是在子目录中默认情况下,
EnumerateFiles
只列出顶级目录中的文件。要列出所有子目录中的所有文件,您需要使用EnumerateFiles
:的重载
Directory.EnumerateFiles(directory, "*.dll", SearchOption.AllDirectories)
您确定您的工作目录是正确的吗?否则Environment.CurrentDirectory
将不会指向您所期望的内容。如果DLL文件与执行代码位于同一目录中,则可以执行以下操作:
Assembly.GetExecutingAssembly().Location
然后使用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互操作,但它更复杂。。。