许多文件扩展名和一个可执行应用程序绑定。正因为这样你才可以使用Invoke-Item打开一个文档。
要找出一个给定后缀名的文件是由那个默认引用程序打开它,并不麻烦。我们可以使用Windows系统中的注册表,自行编程解决。但是在扫描注册表时,要稍微留意一下32位和64位机器的问题,这不是本文重点,点到为止。
另外一种途径,稍显旁门左道,调用Windows API。下面的例子会演示如何调用。采取这种途径最大的优势是借力于操作系统。而你的付出成本只是用C#代码间接调用Windows API中的函数而已:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
|
$Source = @"
using System;
using System.Text;
using System.Runtime.InteropServices;
public class Win32API
{
[DllImport( "shell32.dll" , EntryPoint= "FindExecutable" )]
public static extern long FindExecutableA(string lpFile, string lpDirectory, StringBuilder lpResult);
public static string FindExecutable(string pv_strFilename)
{
StringBuilder objResultBuffer = new StringBuilder(1024);
long lngResult = 0;
lngResult = FindExecutableA(pv_strFilename, string.Empty, objResultBuffer);
if (lngResult >= 32)
{
return objResultBuffer.ToString();
}
return string.Format( "Error: ({0})" , lngResult);
}
}
"@
Add-Type -TypeDefinition $Source -ErrorAction SilentlyContinue
$FullName = 'c:\Windows\windowsupdate.log'
$Executable = [Win32API]::FindExecutable($FullName)
"$FullName will be launched by $Executable"
|
唯一有个限制,就是FindExecutable()需要检查的文件是存在的,你不能只用文件扩展名去请求。
另外@reidca反馈说该方法不能检测MMC加载项打开的文件,比如cer和pfx证书文件,程序会崩溃。