Answers:
我认为您可以可靠地获得的最接近的结果是确定所需的CLR版本。您可以通过使用ILDASM并查看“ MANIFEST”节点或Reflector并以IL形式查看“ Application.exe”节点的分散视图来完成此操作。在这两种情况下,都有注释指示CLR版本。在ILDASM中,注释为“ //元数据版本”,在Reflector中,注释为“目标运行时版本”。
以下是名为WindowsFormsApplication1.exe的.NET WinForms应用程序的示例:
ILDASM:
// Metadata version: v2.0.50727
.assembly extern mscorlib
{
.publickeytoken = (B7 7A 5C 56 19 34 E0 89 ) // .z\V.4..
.ver 2:0:0:0
}
.assembly extern System
{
.publickeytoken = (B7 7A 5C 56 19 34 E0 89 ) // .z\V.4..
.ver 2:0:0:0
}
反光板:
.module WindowsFormsApplication1.exe
.subsystem 0x0002
// MVID: {CA3D2090-16C5-4899-953E-4736D6BC0FA8}
// Target Runtime Version: v2.0.50727
您还可以查看引用的程序集列表,并查找具有最高版本号的引用。
同样,使用ILDASM查看“ MANIFEST”节点数据:
.assembly extern System.Drawing
{
.publickeytoken = (B0 3F 5F 7F 11 D5 0A 3A ) // .?_....:
.ver 2:0:0:0
}
.assembly extern System.Core
{
.publickeytoken = (B7 7A 5C 56 19 34 E0 89 ) // .z\V.4..
.ver 3:5:0:0
}
并使用Reflector,查看列出的每个参考的虚弱(仍为IL):
.assembly extern System.Core
{
.ver 3:5:0:0
.publickeytoken = (B7 7A 5C 56 19 34 E0 89)
}
通过查找具有最高版本元数据的引用,您可以确定引用来自哪个版本的Framework,这表明您需要安装相同版本的Framework才能运行该应用程序。话虽如此,我不会将其视为100%可靠,但我认为它不会很快改变。
使用记事本(已存在三十年了,大小为200kb)预安装了以下工具:
notepad appname.exe
, F3
直到.NET Framework,version=vX.Y
出现v2.
...仍然容易100倍,然后安装了千兆字节的点网分析器工具和垃圾工作室。其他任何编辑器/查看器也可以打开二进制文件,例如Notepad ++或totalCommander的出色文本/十六进制查看器列表器。
现在,您可以使用ILSpy来检查程序集的目标框架。加载程序集后,单击程序集节点的根,您可以在TargetFramework声明下找到信息:
[assembly: TargetFramework(".NETFramework,Version=v4.5", FrameworkDisplayName = ".NET Framework 4.5")]
通过代码可以使用,Assembly.ImageRuntimeVersion
但是通过查看文件,可能最好的方法是使用反射器并查看mscorlib
所引用的版本。
编辑: 更好的方法是使用ildasm,打开程序集,然后查看程序集的清单。清单的第一行将告诉您为程序集构建的CLR 的确切版本。
您可以使用一个名为CorFlags.exe的工具。自.NET 2.0以来,它就已经存在,并且我肯定知道它包含在Windows SDK 7.0中。默认情况下(在Windows XP Pro上)它将安装到C:\ Program Files \ Microsoft SDKs \ Windows \ v7.0A \ bin \ CorFlags.exe。为它提供托管模块的文件路径(不带任何其他命令行标志)以显示其头信息,包括版本。
请记住,此实用程序旨在修改模块的PE32标头,因此在仔细阅读文档之前,请勿使用任何标志。
或者,您可以仅查找System.Core的哪个引用。这将告诉您此应用程序正在使用的.NET Framework版本。对于2.0,System.Core的版本将为2.0.xxx.xxx。对于3.5,版本将是3.5.xxx.xxx,依此类推。
您可以使用Powershell在Windows中获取文件的.NET版本。以下脚本;
$path=’.\’
$ErrorActionPreference = "SilentlyContinue"
$files=Get-ChildItem -Path $path -Recurse -include *.dll,*.exe
foreach($file in $files)
{
$filename = $file.BaseName
$version = $([System.Reflection.Assembly]::ReflectionOnlyLoadFrom($file.FullName).GetCustomAttributesData() |
select-object -ExpandProperty ConstructorArguments |
select-object -ExpandProperty Value |
select-string -Pattern '.NET')
Write-Output "$filename,$version"
}
请注意,该结果为该文件夹中的exe文件提取了.NET版本,但对于dll也将执行相同的操作。