如何使用.NET升级COM组件

我发现了一篇关于如何通过调用CoCreateInstanceAsAdmin来提升用C ++编写的COM对象的文章 。 但是我无法找到或做的是将.NET(c#)应用程序的一个组件实现为COM对象,然后调用该对象来执行需要UAC提升的任务。 MSDN将此文档记录为管理COM对象模型 。

我知道以管理员身份启动应用程序(或其他应用程序)是可能的,并且很容易在单独的过程中执行任务(例如参见Daniel Moth的post ,但我正在寻找的是一种方式从同一个未提升的.NET可执行文件中做所有事情。当然,这样做会在新进程中生成COM对象,但是由于透明的编组,.NET COM对象的调用者不应该(也是很多)意识到这一点。

关于如何使用C#项目,通过CoCreateInstanceAsAdmin API实现用C#编写的COM对象的任何想法都会非常有用。 所以我真的很想学习如何在C#中编写COM对象,然后我可以通过COM高程API从C#调用它。

不要紧,boost的COM对象是否在同一进程中运行。 我只是不想让整个应用程序升级; 我只想提升将执行代码的COM对象。 如果我能写出一些东西:

 // in a dedicated assembly, marked with the following attributes: [assembly: ComVisible (true)] [assembly: Guid ("....")] public class ElevatedClass { public void X() { /* do something */ } } 

然后让我的主应用程序通过CoCreateInstanceAsAdmin调用实例化ElevatedClass 。 但也许我只是在做梦。

查看Windows Vista UAC演示示例代码

(您还需要针对UnsafeNativeMethods.CoGetObject方法的Vista Bridge示例)

这为您提供了C#代码,显示了几种不同的提升方式,包括COM对象

(不完整的代码示例 – 抓取上面的文件)

 [return: MarshalAs(UnmanagedType.Interface)] static internal object LaunchElevatedCOMObject(Guid Clsid, Guid InterfaceID) { string CLSID = Clsid.ToString("B"); // B formatting directive: returns {xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx} string monikerName = "Elevation:Administrator!new:" + CLSID; NativeMethods.BIND_OPTS3 bo = new NativeMethods.BIND_OPTS3(); bo.cbStruct = (uint)Marshal.SizeOf(bo); bo.hwnd = IntPtr.Zero; bo.dwClassContext = (int)NativeMethods.CLSCTX.CLSCTX_ALL; object retVal = UnsafeNativeMethods.CoGetObject(monikerName, ref bo, InterfaceID); return (retVal); } 

提升的要素是过程。 因此,如果我正确理解您的问题,并且您想要一种方法来提升流程中的COM对象,那么答案就是您不能。 CoCreateInstanceAsAdmin的全部内容是不在您的进程中运行它。

我认为CoCreateInstanceAsAdmin工作的唯一方法是提前注册COM组件。 如果您希望应用程序在XCopy部署设置中工作,那么这可能是一个问题。

出于我自己在Gallio的目的,我决定在旁边创建一个小的托管进程,并使用清单来要求管理员权限。 然后,当我需要执行提升操作时,我启动托管进程的一个实例,并通过.Net remoting指示它执行在Gallio的Inversion of Control容器中注册的特定命令。

这是一项相当多的工作,但Gallio已经有一个进程外托管设施,因此在混合中增加高度并不是太难。 此外,此机制确保Gallio可以执行权限提升,而无需事先在注册表中安装任何其他COM组件。