如何从 PowerShell 中的 COM 对象获取不同的接口

How to obatin a different interface from a COM object in PowerShell

使用像 [System.Runtime.InteropServices.Marshal]::GetActiveObject("VisualStudio.DTE") 这样的 COM 方法,我可以很好地导航 Visual Studio DTE 对象模型。例如来自 DTE object I can get Debugger, and then LocalProcesses, and the the Process object. But I need the derived Process2 interface on it, to call Attach2("<my debug engine>")。 我找不到获取所需接口的方法,简单的转换会导致运行时错误:Cannot convert the "System.__ComObject" value of type "System.__ComObject#{5c5a0070-f396-4e37-a82a-1b767e272df9}" to type "EnvDTE80.Process2".

PS> $dte = [System.Runtime.InteropServices.Marshal]::GetActiveObject("VisualStudio.DTE")
PS> $p = $dte.Debugger.LocalProcesses | where {$_.ProcessID -eq 11212}
PS> $p


Name       : C:\Program Files\IIS Express\iisexpress.exe
ProcessID  : 11212
Programs   : System.__ComObject
DTE        : System.__ComObject
Parent     : System.__ComObject
Collection : System.__ComObject



PS> [EnvDTE80.Process2]$p2 = $p
Cannot convert the "System.__ComObject" value of type "System.__ComObject#{5c5a0070-f396-4e37-a82a-1b767e272df9}" to type "EnvDTE80.Process2".
At line:1 char:1
+ [EnvDTE80.Process2]$p2 = $p
+ ~~~~~~~~~~~~~~~~~~~~~~~~~~~
    + CategoryInfo          : MetadataError: (:) [], ArgumentTransformationMetadataException
    + FullyQualifiedErrorId : RuntimeException

你真的不能,至少不能以 PowerShell 在涉及成员绑定时能够记住的方式。

PowerShell 仅基于运行时信息运行。即使您首先在 C# 中转换它,如果 QueryInterface returns 该接口的相同指针,那么所有 PowerShell 将看到的是它当前检测到的 IDispatch。即使您获得的对象是来自主要互操作程序集的强类型版本,PowerShell 也只能看到具体类型(Process2 似乎没有)。

作为解决方法,您可以使用反射:

[EnvDTE80.Process2].InvokeMember(
    'Attach2',
    [Reflection.BindingFlags]::InvokeMethod,
    <# binder: #> $null,
    <# target: #> $process,
    <# args: #> @($myEngine))