草庐IT

c# - 从通过模态 WinForms 调用的 C++ 调用 IConnectionPointImpl 接口(interface)时出现问题

coder 2024-02-04 原文

我们有一个 native C++ 应用程序,它支持一些基于 COM 的各种类型的 VBA 宏。这些类型之一,VBAExtension , 将自身注册到核心 C++ 应用程序,从而产生(派生自的类)IConnectionPointImpl<Extension, &DIID_IExtensionEvents, CComDynamicUnkArray> 的实例.这很好用;给定适当的 VBAExtension 对象,核心和其他 VBA 宏都可以访问 IExtensionEvents 上的方法。

我们还有一个 .NET 程序集(用 C# 编写),它也在运行时加载到核心应用程序中。由于历史原因,程序集由自动运行的 VBA 宏加载;然后,当用户按下特定按钮时,另一个 VBA 宏将运行程序集的主入口点,这将显示 System.Windows.Forms。用于进一步交互的对话框。

这就是设置。我在访问 VBAExtension 时看到一些奇怪的行为.NET 程序集中的方法。具体来说,我从程序集中的不同位置运行以下代码:

foreach (VBAExtension ve in app.Extensions)
{
    System.Diagnostics.Debug.Print("Ext: " + ve.Name);
}

如果我从程序集的主要对象的构造函数中运行它;或者从程序集的主入口点(在显示对话框之前),一切都很好——我得到了 VBAExtension 的名称打印出来了。

但是,如果我从程序集(模态 - 我们正在调用 form.ShowDialog())WinForm 中的按钮启动的命令运行相同的代码,ve.Name都是空白。 pDispatch->Invoke IConnectionPointImpl 打来的电话子类成功(返回 S_OK),但未设置任何返回变量。

如果我将对话框更改为非模态(使用 form.Show() 调用),那么名称将再次起作用。形式的模态(模态?)似乎会影响 IConnectionPointImpl调用成功。

有人知道这是怎么回事吗?

编辑:自第一次发帖以来,我已经证明重要的不是调用堆栈;相反,它是调用是否来自模式对话框。我已经更新了正文。

编辑 2:Per Hans Passant 的回答,以下是他的诊断问题的答案:

  • 正如预期的那样,在良好(无模式)情况下,如果我重命名 VBA 事件处理程序,则不会出现错误。该调用不返回任何数据。
  • 我已将 MsgBox 调用放入 VBA 处理程序;它在无模式情况下显示,但在模式情况下不显示。因此,处理程序不会在模态情况下执行。
  • 通过使用 Err ,我可以看出,如果我们在 VBA 处理程序中遇到异常,我们会得到一个 VBA 错误对话框。一旦清除它,C++ Invoke调用具有 0x80020009(“发生异常”)作为返回码,并且 pExcepInfo 填充了通用故障值(VBA 吞没了实际细节)
  • 该事件不会在模态对话框的第二次显示时触发,无论是紧随第一个对话框还是在第二次调用 C# 插件期间。

下一步我将尝试深入研究我们的消息循环。

最佳答案

这个问题中几乎没有确凿的事实可以作为答案。可能是非常简单的事情,可能是一个令人讨厌的内存损坏问题或 VBA 解释器内部对线程状态的模糊依赖。粗略的诊断是 VBA 事件处理程序根本没有运行。这在一般情况下并不少见,Basic 中用于声明事件处理程序的声明式风格几乎没有留下诊断订阅问题的好方法。许多 VBA 程序员在尝试解决“为什么事件处理程序未运行”这样的问题时都头晕目眩。

首先收集一些确凿的事实并将它们添加到您的问题中:

  • 首先验证您的 C++ 代码是否确实可以看到根本没有事件处理程序。使用好的版本,重命名事件处理程序。期望是您不这样做,引发接收器未订阅的事件不是错误。
  • 验证事件处理程序是否在错误版本中实际执行。让它做一些除了分配 BSTR 参数之外的事情,有些你可以很容易地看到磁盘上的文件。
  • 验证您是否可以正确诊断事件处理程序中的异常。分配 Err 对象并验证您的 C++ 代码是否生成正确的诊断。请注意,您的 IDispatch::Invoke() 调用为 pExcepInfo 传递了 NULL,这不是生成诊断的好方法。
  • 检查事件是否在您显示窗口时第二次运行,如果是,则说明执行顺序有问题。

稍微关注一下 ShowDialog()。这种方法确实有很多副作用。不再起作用的第一件事是 C++ 代码中的消息循环。现在是分派(dispatch)消息的 .NET 消息循环。查看您的副作用,比简单的 GetMessage/DispatchMessage() 做更多的工作。这项工作不再完成。还要在您的代码库中搜索 PostThreadMessage(),这些消息会在 .NET 代码运行时掉落在地板上。

请记住,当 C# 代码调用 ShowDialog() 时,您的 native C++ 代码将失去控制。在您关闭窗口之前,它不会重新获得控制权。这可能会触发一个简单的执行顺序问题,您的 C++ 代码在执行使 C# 代码运行的所有操作后不应执行任何重要操作。

关于c# - 从通过模态 WinForms 调用的 C++ 调用 IConnectionPointImpl 接口(interface)时出现问题,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/32997005/

有关c# - 从通过模态 WinForms 调用的 C++ 调用 IConnectionPointImpl 接口(interface)时出现问题的更多相关文章

  1. ruby - ECONNRESET (Whois::ConnectionError) - 尝试在 Ruby 中查询 Whois 时出错 - 2

    我正在用Ruby编写一个简单的程序来检查域列表是否被占用。基本上它循环遍历列表,并使用以下函数进行检查。require'rubygems'require'whois'defcheck_domain(domain)c=Whois::Client.newc.query("google.com").available?end程序不断出错(即使我在google.com中进行硬编码),并打印以下消息。鉴于该程序非常简单,我已经没有什么想法了-有什么建议吗?/Library/Ruby/Gems/1.8/gems/whois-2.0.2/lib/whois/server/adapters/base.

  2. ruby - 在 64 位 Snow Leopard 上使用 rvm、postgres 9.0、ruby 1.9.2-p136 安装 pg gem 时出现问题 - 2

    我想为Heroku构建一个Rails3应用程序。他们使用Postgres作为他们的数据库,所以我通过MacPorts安装了postgres9.0。现在我需要一个postgresgem并且共识是出于性能原因你想要pggem。但是我对我得到的错误感到非常困惑当我尝试在rvm下通过geminstall安装pg时。我已经非常明确地指定了所有postgres目录的位置可以找到但仍然无法完成安装:$envARCHFLAGS='-archx86_64'geminstallpg--\--with-pg-config=/opt/local/var/db/postgresql90/defaultdb/po

  3. ruby - 通过 rvm 升级 ruby​​gems 的问题 - 2

    尝试通过RVM将RubyGems升级到版本1.8.10并出现此错误:$rvmrubygemslatestRemovingoldRubygemsfiles...Installingrubygems-1.8.10forruby-1.9.2-p180...ERROR:Errorrunning'GEM_PATH="/Users/foo/.rvm/gems/ruby-1.9.2-p180:/Users/foo/.rvm/gems/ruby-1.9.2-p180@global:/Users/foo/.rvm/gems/ruby-1.9.2-p180:/Users/foo/.rvm/gems/rub

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

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

  5. ruby - 通过 erb 模板输出 ruby​​ 数组 - 2

    我正在使用puppet为ruby​​程序提供一组常量。我需要提供一组主机名,我的程序将对其进行迭代。在我之前使用的bash脚本中,我只是将它作为一个puppet变量hosts=>"host1,host2"我将其提供给bash脚本作为HOSTS=显然这对ruby​​不太适用——我需要它的格式hosts=["host1","host2"]自从phosts和putsmy_array.inspect提供输出["host1","host2"]我希望使用其中之一。不幸的是,我终其一生都无法弄清楚如何让它发挥作用。我尝试了以下各项:我发现某处他们指出我需要在函数调用前放置“function_”……这

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

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

  7. ruby - 通过 RVM (OSX Mountain Lion) 安装 Ruby 2.0.0-p247 时遇到问题 - 2

    我的最终目标是安装当前版本的RubyonRails。我在OSXMountainLion上运行。到目前为止,这是我的过程:已安装的RVM$\curl-Lhttps://get.rvm.io|bash-sstable检查已知(我假设已批准)安装$rvmlistknown我看到当前的稳定版本可用[ruby-]2.0.0[-p247]输入命令安装$rvminstall2.0.0-p247注意:我也试过这些安装命令$rvminstallruby-2.0.0-p247$rvminstallruby=2.0.0-p247我很快就无处可去了。结果:$rvminstall2.0.0-p247Search

  8. ruby-on-rails - Enumerator.new 如何处理已通过的 block ? - 2

    我在理解Enumerator.new方法的工作原理时遇到了一些困难。假设文档中的示例:fib=Enumerator.newdo|y|a=b=1loopdoy[1,1,2,3,5,8,13,21,34,55]循环中断条件在哪里,它如何知道循环应该迭代多少次(因为它没有任何明确的中断条件并且看起来像无限循环)? 最佳答案 Enumerator使用Fibers在内部。您的示例等效于:require'fiber'fiber=Fiber.newdoa=b=1loopdoFiber.yieldaa,b=b,a+bendend10.times.m

  9. 使用 ACL 调用 upload_file 时出现 Ruby S3 "Access Denied"错误 - 2

    我正在尝试编写一个将文件上传到AWS并公开该文件的Ruby脚本。我做了以下事情:s3=Aws::S3::Resource.new(credentials:Aws::Credentials.new(KEY,SECRET),region:'us-west-2')obj=s3.bucket('stg-db').object('key')obj.upload_file(filename)这似乎工作正常,除了该文件不是公开可用的,而且我无法获得它的公共(public)URL。但是当我登录到S3时,我可以正常查看我的文件。为了使其公开可用,我将最后一行更改为obj.upload_file(file

  10. ruby - 寻找通过阅读代码确定编程语言的ruby gem? - 2

    几个月前,我读了一篇关于ruby​​gem的博客文章,它可以通过阅读代码本身来确定编程语言。对于我的生活,我不记得博客或gem的名称。谷歌搜索“ruby编程语言猜测”及其变体也无济于事。有人碰巧知道相关gem的名称吗? 最佳答案 是这个吗:http://github.com/chrislo/sourceclassifier/tree/master 关于ruby-寻找通过阅读代码确定编程语言的rubygem?,我们在StackOverflow上找到一个类似的问题:

随机推荐