c# – VSIX中令人讨厌的COM互操作问题
一段时间以来,我在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,如果我看过一个. 无论如何,我将坚持这一点,直到我找到适当的解决方案或再次出现错误. (编辑:李大同) 【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容! |