草庐IT

c++ - 如何在进程外客户端中获取免注册 COM 对象代理

coder 2024-06-18 原文

我不知道是否有使这成为可能的 API,或者我是否必须自己动手。这就是我想要完成的。

我有一个连接到 NT 服务以启动与另一个 COM 服务器的 session 的应用程序。

  • 应用程序,客户端。
  • 经纪人 NT 服务; (系统帐户上下文)。
  • session COM 服务; (系统帐户上下文,将根据需要模拟用户)。

session 服务器将为连接到 NT 服务的每个应用程序实例都有一个运行实例。应用程序可以请求 session 服务器加载 COM 库 DLL,并从 session 服务器中的那些 DLL 托管对象和服务。 DLL 通过免注册激活进行注册。

从 session 服务器创建对象并将它们传回应用程序只要它们是 IDispatch 派生的就可以正常工作,这是整个系统的要求,因为脚本语言可能会使用它,这就是请求的接口(interface). C++ 应用程序还可以使用 session 服务器中托管的对象。但是 IDispatch 是一个在 C++ 中处理的过于冗长的接口(interface)。

我的问题是:

鉴于托管的 DLL 具有应用程序确实知道的双重自定义接口(interface),并且应用程序可以通过 ITypeInfo 读取有关这些接口(interface)的类型信息;如果我可以为它提供 IDispatch 接口(interface),是否有一个 API 在运行时会创建一个代理来模仿原始的自定义接口(interface),该接口(interface)还带有 ITypeInfo 信息。所有代理需要的是调用 IDispatch 接口(interface),但对 C++ 来说似乎是自定义接口(interface)。更优化的解决方案是使用相同的代理,即默认的 OLE 自动化代理,DLL 在其 list 中注册。

我无法为 DLL 注册代理/ stub ,因为多个应用程序可能具有相同的模块,但版本不同,因此使用免注册激活。

最佳答案

类型库中描述的任何 [oleautomation] 接口(interface)(和任何 [dual] 接口(interface))都可以使用类型库编码(marshal)拆收器。

在这里,您可以将查找代理 stub DLL 的麻烦换成查找类型库的麻烦。因此,您在程序集的 list 中(直接在 assembly 元素下)声明接口(interface)和类型库,如下所示:

<comInterfaceExternalProxyStub name="IFooBar"
                               iid="{IIIIIIII-IIII-IIII-IIII-IIIIIIIIIIII}"
                               proxyStubClsid32="{00020424-0000-0000-C000-000000000046}"
                               tlbid="{TTTTTTTT-TTTT-TTTT-TTTT-TTTTTTTTTTTT}" />

<!-- This also works for a type library embedded in a DLL -->
<file name="FooBar.tlb">
    <!-- If you have multiple embedded type libraries, use the resourceid attribute -->
    <typelib tlbid="{TTTTTTTT-TTTT-TTTT-TTTT-TTTTTTTTTTTT}"
             version="1.0" />
</file>

如果您的界面不是[oleautomation] 界面,并且您想要隔离代理 stub DLL,您可以使用如下内容:

<file name="FooBarPS.dll">
    <comInterfaceProxyStub name="IFooBar"
                           iid="{IIIIIIII-IIII-IIII-IIII-IIIIIIIIIIII}"
                           proxyStubClsid32="{PPPPPPPP-PPPP-PPPP-PPPP-PPPPPPPPPPPP}"
                           threadingModel="Both" />
</file>

comInterfaceProxyStub 很像 comClass,但专注于代理/ stub 并且它与接口(interface)相关联。

您可以使用一对 comInterfaceExternalProxyStub(在 assembly 元素级别)和 comClass(在 file 元素),如果您想使用和不使用隔离的代理/ stub DLL 进行测试,通过(取消)注释代理/ stub file 部分:

<comInterfaceExternalProxyStub name="IFooBar"
                               iid="{IIIIIIII-IIII-IIII-IIII-IIIIIIIIIIII}"
                               proxyStubClsid32="{PPPPPPPP-PPPP-PPPP-PPPP-PPPPPPPPPPPP}" />

<file name="FooBarPS.dll">
    <comClass description="PSFactoryBuffer"
              clsid="{PPPPPPPP-PPPP-PPPP-PPPP-PPPPPPPPPPPP}"
              threadingModel="Both" />
</file>

我不确定,但如果您的标准代理/ stub DLL 用于多个接口(interface),您也必须使用这种方法。


编辑:看起来这些对您来说都不是新的。您的问题是,在 session 服务中,即使您激活了动态加载库的 list ,该状态也只保留在当前线程中。因此,COM 工作线程(例如 RPC 线程)将不知道您的接口(interface)、组件类和代理/ stub 。

您在代理服务 (REGDB_E_IIDNOTREG) 中看到的错误可能源于从 session 服务返回的编码。您不会在 session 服务中遇到该错误,因为它发生在您的方法返回之后

但是,它可能发生在代理服务中,因为它很自然:它不加载任何库,更不用说它们的 list 了。

我建议您采用的方法是使 session 服务和代理服务具有依赖于声明免注册 COM 信息的程序集的 list 。这样,它就会成为默认激活上下文的一部分,您无需对激活上下文执行任何操作。

请记住,您无法控制不属于您的线程的激活上下文,除非让默认激活上下文包含您预先需要的内容。

关于你的这部分问题:

I cannot register the proxy/stubs for the DLLs since multiple application may have the same modules, but differing in version, hence the use of registration-free activation.

我不清楚你想说什么。如果您的模块向后兼容,则无需担心。如果不是,请使用不同的 CLSID/ProgID。

我希望您并不是说您使用的是具有实际不同接口(interface)的相同 IID,因为这违反了 COM。解决这个问题的最好方法就是不这样做,期间。另一种方法是在 session 服务 代理服务中使用具有专用激活上下文的专用线程,正如您可能只在 session 服务中看到的那样,这是一种非常脆弱的方法。

在我看来,您可能根本不需要 COM 隔离。但是,如果您仍然需要它,则需要对服务、代理和 session 都执行此操作,并从它们的 list 中执行,而不是在运行时执行。

关于c++ - 如何在进程外客户端中获取免注册 COM 对象代理,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/31753648/

有关c++ - 如何在进程外客户端中获取免注册 COM 对象代理的更多相关文章

  1. ruby - 如何在 Ruby 中顺序创建 PI - 2

    出于纯粹的兴趣,我很好奇如何按顺序创建PI,而不是在过程结果之后生成数字,而是让数字在过程本身生成时显示。如果是这种情况,那么数字可以自行产生,我可以对以前看到的数字实现垃圾收集,从而创建一个无限系列。结果只是在Pi系列之后每秒生成一个数字。这是我通过互联网筛选的结果:这是流行的计算机友好算法,类机器算法:defarccot(x,unity)xpow=unity/xn=1sign=1sum=0loopdoterm=xpow/nbreakifterm==0sum+=sign*(xpow/n)xpow/=x*xn+=2sign=-signendsumenddefcalc_pi(digits

  2. ruby - 如何在 buildr 项目中使用 Ruby 代码? - 2

    如何在buildr项目中使用Ruby?我在很多不同的项目中使用过Ruby、JRuby、Java和Clojure。我目前正在使用我的标准Ruby开发一个模拟应用程序,我想尝试使用Clojure后端(我确实喜欢功能代码)以及JRubygui和测试套件。我还可以看到在未来的不同项目中使用Scala作为后端。我想我要为我的项目尝试一下buildr(http://buildr.apache.org/),但我注意到buildr似乎没有设置为在项目中使用JRuby代码本身!这看起来有点傻,因为该工具旨在统一通用的JVM语言并且是在ruby中构建的。除了将输出的jar包含在一个独特的、仅限ruby​​

  3. ruby - 什么是填充的 Base64 编码字符串以及如何在 ruby​​ 中生成它们? - 2

    我正在使用的第三方API的文档状态:"[O]urAPIonlyacceptspaddedBase64encodedstrings."什么是“填充的Base64编码字符串”以及如何在Ruby中生成它们。下面的代码是我第一次尝试创建转换为Base64的JSON格式数据。xa=Base64.encode64(a.to_json) 最佳答案 他们说的padding其实就是Base64本身的一部分。它是末尾的“=”和“==”。Base64将3个字节的数据包编码为4个编码字符。所以如果你的输入数据有长度n和n%3=1=>"=="末尾用于填充n%

  4. ruby-on-rails - 如何在 ruby​​ 中使用两个参数异步运行 exe? - 2

    exe应该在我打开页面时运行。异步进程需要运行。有什么方法可以在ruby​​中使用两个参数异步运行exe吗?我已经尝试过ruby​​命令-system()、exec()但它正在等待过程完成。我需要用参数启动exe,无需等待进程完成是否有任何ruby​​gems会支持我的问题? 最佳答案 您可以使用Process.spawn和Process.wait2:pid=Process.spawn'your.exe','--option'#Later...pid,status=Process.wait2pid您的程序将作为解释器的子进程执行。除

  5. ruby-on-rails - 如何优雅地重启 thin + nginx? - 2

    我的瘦服务器配置了nginx,我的ROR应用程序正在它们上运行。在我发布代码更新时运行thinrestart会给我的应用程序带来一些停机时间。我试图弄清楚如何优雅地重启正在运行的Thin实例,但找不到好的解决方案。有没有人能做到这一点? 最佳答案 #Restartjustthethinserverdescribedbythatconfigsudothin-C/etc/thin/mysite.ymlrestartNginx将继续运行并代理请求。如果您将Nginx设置为使用多个上游服务器,例如server{listen80;server

  6. ruby - 在 jRuby 中使用 'fork' 生成进程的替代方案? - 2

    在MRIRuby中我可以这样做:deftransferinternal_server=self.init_serverpid=forkdointernal_server.runend#Maketheserverprocessrunindependently.Process.detach(pid)internal_client=self.init_client#Dootherstuffwithconnectingtointernal_server...internal_client.post('somedata')ensure#KillserverProcess.kill('KILL',

  7. ruby - 如何在续集中重新加载表模式? - 2

    鉴于我有以下迁移:Sequel.migrationdoupdoalter_table:usersdoadd_column:is_admin,:default=>falseend#SequelrunsaDESCRIBEtablestatement,whenthemodelisloaded.#Atthispoint,itdoesnotknowthatusershaveais_adminflag.#Soitfails.@user=User.find(:email=>"admin@fancy-startup.example")@user.is_admin=true@user.save!ende

  8. ruby - 通过 ruby​​ 进程共享变量 - 2

    我正在编写一个gem,我必须在其中fork两个启动两个webrick服务器的进程。我想通过基类的类方法启动这个服务器,因为应该只有这两个服务器在运行,而不是多个。在运行时,我想调用这两个服务器上的一些方法来更改变量。我的问题是,我无法通过基类的类方法访问fork的实例变量。此外,我不能在我的基类中使用线程,因为在幕后我正在使用另一个不是线程安全的库。所以我必须将每个服务器派生到它自己的进程。我用类变量试过了,比如@@server。但是当我试图通过基类访问这个变量时,它是nil。我读到在Ruby中不可能在分支之间共享类变量,对吗?那么,还有其他解决办法吗?我考虑过使用单例,但我不确定这是

  9. ruby - 如何在 Ruby 中拆分参数字符串 Bash 样式? - 2

    我正在为一个项目制作一个简单的shell,我希望像在Bash中一样解析参数字符串。foobar"helloworld"fooz应该变成:["foo","bar","helloworld","fooz"]等等。到目前为止,我一直在使用CSV::parse_line,将列分隔符设置为""和.compact输出。问题是我现在必须选择是要支持单引号还是双引号。CSV不支持超过一个分隔符。Python有一个名为shlex的模块:>>>shlex.split("Test'helloworld'foo")['Test','helloworld','foo']>>>shlex.split('Test"

  10. ruby - 如何在 Lion 上安装 Xcode 4.6,需要用 RVM 升级 ruby - 2

    我实际上是在尝试使用RVM在我的OSX10.7.5上更新ruby,并在输入以下命令后:rvminstallruby我得到了以下回复:Searchingforbinaryrubies,thismighttakesometime.Checkingrequirementsforosx.Installingrequirementsforosx.Updatingsystem.......Errorrunning'requirements_osx_brew_update_systemruby-2.0.0-p247',pleaseread/Users/username/.rvm/log/138121

随机推荐