加入收藏 | 设为首页 | 会员中心 | 我要投稿 李大同 (https://www.lidatong.com.cn/)- 科技、建站、经验、云计算、5G、大数据,站长网!
当前位置: 首页 > 百科 > 正文

c# – VSIX中令人讨厌的COM互操作问题

发布时间:2020-12-15 21:21:00 所属栏目:百科 来源:网络整理
导读:一段时间以来,我在Visual Studio 2010的VSIX包中发现了间歇性的COM问题.尝试订阅IDE的基于COM的事件接收器之一会随机抛出以下错误: “无法使用已与其基础RCW分离的COM对象” 一个repro案例归结为这个代码(显然必须在VSIX中使用): using System;using EnvDT
一段时间以来,我在Visual Studio 2010的VSIX包中发现了间歇性的COM问题.尝试订阅IDE的基于COM的事件接收器之一会随机抛出以下错误:

“无法使用已与其基础RCW分离的COM对象”

一个repro案例归结为这个代码(显然必须在VSIX中使用):

using System;
using EnvDTE;
using EnvDTE80;

class Test
{
    private readonly Events _events;
    private readonly Events2 _events2;
    private readonly BuildEvents _buildEvents;
    private readonly ProjectItemsEvents _projectItemsEvents;

    public Test(IServiceProvider provider)
    {
        var dte = (DTE)provider.GetService(typeof(DTE));
        var dte2 = (DTE2)dte;

        // Store all references in fields as a GC precaution.
        _events = dte.Events;
        _events2 = (Events2)dte2.Events;
        _buildEvents = _events.BuildEvents;
        _projectItemsEvents = _events2.ProjectItemsEvents;

        // Proceed to subscribe to event sinks.
        _buildEvents.OnBuildBegin += BuildBeginHandler; // BOOM!
        _projectItemsEvents.ItemAdded += ItemAddedHandler;
    }

    private void ItemAddedHandler(ProjectItem projectItem) { }

    private void BuildBeginHandler(vsBuildScope scope,vsBuildAction action) { }
}

我已经从网上可以找到的类似问题的大量描述中了解到可能的原因.它基本上是在COM互操作期间Runtime Callable Wrappers和GC交互方式的副作用.这是一个类似的问题link完成解释.

我对这个解释很好,特别是因为它提示了一个简单的解决方法 – 将事件接收器引用存储在一个字段中,以防止它过早地GC.实际上,许多人似乎已经以这种方式解决了他们的问题.

困扰我的是它在我的情况下不起作用.我真的很难过为什么.正如您可以清楚地看到的那样,我已经将所有对象引用存储在字段中作为预防措施.但错误仍然存??在.我试着在ctor结束时使用GC.KeepAlive()调用更加明确,但无济于事.还有什么要做的吗?

如果没有解决方案,我的VSIX随机无法加载,只为用户提供一个选项:重启Visual Studio并希望下次不会发生.

任何帮助将真正受到赞赏!

解决方法

好吧,我放弃了,只是做了我脑子里唯一想到的事情.我认为,由于这显然是一种竞争条件,我不能以可预测的方式影响,如果我输了,我不妨重新参加比赛.

所以我将订阅行移动到了一个while循环中,尝试…捕获它们并在一些Thread.Sleep()之后重试.当两个订阅成功或者我连续失去比赛超过2秒时,循环退出.

踢球者是,自从我实施改变以来,我没有输过一次比赛.真正的Heisenbug,如果我看过一个.

无论如何,我将坚持这一点,直到我找到适当的解决方案或再次出现错误.

(编辑:李大同)

【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容!

    推荐文章
      热点阅读