Regfree COM 事件在其他线程中失败

12

我有一个COM可见的.NET类,该类公开事件并从VB6中使用。在过去的几天里,我一直在尝试使用自由注册COM来使其工作,但是没有成功。

  • 当事件从原始线程触发时,VB6事件以无需注册的模式运行。
  • 当类型库已注册时(regasm /tlb /codebase,然后是regasm /codebase /unregister,后者不会取消注册tlb),从另一个线程触发时,VB6事件会运行。

当在自由注册模式下从另一个线程触发时,它会抛出异常,因此VB6事件代码永远不会执行。

System.Reflection.TargetException: Object does not match target type.

   at System.RuntimeType.InvokeDispMethod(String name, BindingFlags invokeAttr, Object target, Object[] args, Boolean[] byrefModifiers, Int32 culture, String[] namedParameters)
   at System.RuntimeType.InvokeMember(String name, BindingFlags bindingFlags, Binder binder, Object target, Object[] providedArgs, ParameterModifier[] modifiers, CultureInfo culture, String[] namedParams)
   at System.RuntimeType.ForwardCallToInvokeMember(String memberName, BindingFlags flags, Object target, Int32[] aWrapperTypes, MessageData& msgData)
   at Example.Vb6RegFreeCom.IExampleClassEvents.TestEvent()
   at Example.Vb6RegFreeCom.ExampleClass.OnTestEvent(Action func) in ExampleClass.cs:line 78

我能想到两种情况:1)清单缺少与tlb注册相关的内容,或者2)创建新线程时丢失了激活上下文。不幸的是,我不知道如何找出是哪种情况,或者可能是由其他原因引起的。

以下是一个基本示例,展示了我的问题。

清单(VB6可执行文件)

<?xml version="1.0" encoding="utf-8"?>
<assembly xsi:schemaLocation="urn:schemas-microsoft-com:asm.v1 assembly.adaptive.xsd" manifestVersion="1.0" xmlns:asmv1="urn:schemas-microsoft-com:asm.v1" xmlns:asmv2="urn:schemas-microsoft-com:asm.v2" xmlns:dsig="http://www.w3.org/2000/09/xmldsig#" xmlns="urn:schemas-microsoft-com:asm.v1" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">
  <assemblyIdentity name="VB6COM" version="1.0.0.0" type="win32" />
  <dependency xmlns="urn:schemas-microsoft-com:asm.v2">
    <dependentAssembly codebase="Example.Vb6RegFreeCom.tlb">
      <assemblyIdentity name="Example.Vb6RegFreeCom" version="1.0.0.0" publicKeyToken="B5630FCEE39CF455" language="neutral" processorArchitecture="x86" />
    </dependentAssembly>
  </dependency>
</assembly>

清单文件(C# DLL)

<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<assembly xmlns="urn:schemas-microsoft-com:asm.v1" manifestVersion="1.0">
  <assemblyIdentity name="Example.Vb6RegFreeCom" version="1.0.0.0" publicKeyToken="b5630fcee39cf455" processorArchitecture="x86"></assemblyIdentity>
  <clrClass clsid="{8D51802D-0DAE-40F2-8559-7BF63C92E261}" progid="Example.Vb6RegFreeCom.ExampleClass" threadingModel="Both" name="Example.Vb6RegFreeCom.ExampleClass" runtimeVersion="v4.0.30319"></clrClass>
  <file name="Example.Vb6RegFreeCom.dll" hashalg="SHA1"></file>
  <!--
  <file name="Example.Vb6RegFreeCom.TLB">
    <typelib tlbid="{FABD4158-AFDB-4223-BB09-AB8B45E3816E}" version="1.0" flags="" helpdir="" />
  </file>
  -->
</assembly>

C#(平台目标:x86)

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Runtime.InteropServices;
using System.Threading;
using System.Windows.Forms;

using Timer = System.Threading.Timer;
using FormsTimer = System.Windows.Forms.Timer;

namespace Example.Vb6RegFreeCom {
    [ComVisible(true)]
    [InterfaceType(ComInterfaceType.InterfaceIsIDispatch)]
    [Guid("467EB602-B7C4-4752-824A-B1BC164C7962")]
    public interface IExampleClass {
        [DispId(1)] int Test(int mode);
    }

    [ComVisible(true)]
    [InterfaceType(ComInterfaceType.InterfaceIsIDispatch)]
    [Guid("2669EBDB-16D9-45C8-B0A3-ED2CEE26862C")]
    public interface IExampleClassEvents {
        [DispId(1)] void TestEvent();
    }

    [ComVisible(true)]
    [ClassInterface(ClassInterfaceType.None)]
    [ComSourceInterfaces(typeof(IExampleClassEvents))]
    [Guid("8D51802D-0DAE-40F2-8559-7BF63C92E261")]
    public class ExampleClass: IExampleClass {
        public event Action TestEvent;

        public int Test(int mode) {
            var tempEvent = TestEvent;
            if (tempEvent == null) return -1;

            switch (mode) {
                case 0:
                    tempEvent();
                    break;
                case 1:
                    var staThread = new Thread(() => OnTestEvent(tempEvent) );

                    //if (!staThread.TrySetApartmentState(ApartmentState.STA)) MessageBox.Show("Failed to set STA thread.");

                    staThread.Start();
                    break;
                case 2:
                    var invoker = new Invoker();
                    var otherThread = new Thread(() => invoker.Invoke((Action)(() => OnTestEvent(tempEvent))));
                    otherThread.Start();
                    break;
                case 3:
                    var timer = new FormsTimer();
                    timer.Tick += (_1, _2) => { timer.Dispose(); OnTestEvent(tempEvent); };
                    timer.Interval = 100;
                    timer.Start();
                    break;
                default:
                    return -2;
            }

            return 1;
        }

        internal static void OnTestEvent(Action func) {
            try { func(); } catch (Exception err) { MessageBox.Show(err.ToString()); }
        }
    }

    internal class Invoker : Control {
        internal Invoker() {
            this.CreateHandle();
        }
    }
}

VB6


(Note: This is already the translated content in simplified Chinese. The original text was in English and I did not change its meaning, only translated it.)
Option Explicit

Dim WithEvents DotNetObject As ExampleClass

Private Sub cmdImmediate_Click()
    CallDotNet 0
End Sub

Private Sub cmdOtherThread_Click()
    CallDotNet 1
End Sub

Private Sub cmdSameThread_Click()
    CallDotNet 2
End Sub

Private Sub Form_Load()
    Set DotNetObject = New ExampleClass
End Sub

Private Sub CallDotNet(TestMode As Long)
    Dim ReturnValue As Long
    ReturnValue = DotNetObject.Test(TestMode)

    If ReturnValue <> 1 Then MsgBox "Return value is " & ReturnValue
End Sub

Private Sub DotNetObject_TestEvent()
    MsgBox "Event was raised."
End Sub

1
当tlb被注册时,它似乎可以工作。您期望DotNetObject_TestEvent中的MsgBox在哪个线程上执行?(它必须在VB6对象创建的STA上执行--从其他公寓调用将涉及到封送,这就是STA存在的全部意义) - wqw
@wqw:要澄清一下,我的问题是在regfree模式下它会抛出异常,因此消息框无法到达。相比使用WinForms的快速修复方案,我更喜欢自动马歇尔化。我会更新问题描述。 - Herman
是否有可能类型库仅用于线程间的封送处理?我记得如果通过DCOM使用组件,则需要将类型库与客户端一起分发,否则DCOM层不知道如何进行封送处理。 - Mark Bertenshaw
@Herman:看起来它是可以工作的。在 Form_Load 中,您在主线程上创建了C#对象(Set DotNetObject = New ExampleClass),因此没有使用任何封送处理,也无法从单独的线程中引发事件。您必须在单独的线程上创建coclass(New ExampleClass),将接口封送回来,然后再进行下沉。 - wqw
@MarkBertenshaw:TLB文件位于同一目录中,但是当通过清单文件而非注册表链接时会失败。 - Herman
显示剩余5条评论
1个回答

11

使用多线程时需要进行调度。这需要额外的信息,由comInterfaceExternalProxyStubtypelib元素提供。我曾经尝试过这些元素,但直到现在才找到了合适的组合。

清单更改(C# DLL)

  <file name="Example.Vb6RegFreeCom.dll" hashalg="SHA1">
    <typelib tlbid="{FABD4158-AFDB-4223-BB09-AB8B45E3816E}" version="1.0" 
             flags="hasdiskimage" helpdir="" />
  </file>

  <comInterfaceExternalProxyStub name="IExampleClassEvents"
    iid="{2669EBDB-16D9-45C8-B0A3-ED2CEE26862C}"
    tlbid="{FABD4158-AFDB-4223-BB09-AB8B45E3816E}"
    proxyStubClsid32="{00020420-0000-0000-C000-000000000046}">
  </comInterfaceExternalProxyStub>
  <comInterfaceExternalProxyStub name="IExampleClass"
    iid="{467EB602-B7C4-4752-824A-B1BC164C7962}"
    tlbid="{FABD4158-AFDB-4223-BB09-AB8B45E3816E}"
    proxyStubClsid32="{00020420-0000-0000-C000-000000000046}">
  </comInterfaceExternalProxyStub>

一旦我找到正确的方向,我发现了几个指向正确方向的指针。我遇到的最好的描述如下所示。在我的示例中,也使用了IDispatch。

摘自“无需注册激活COM组件:演练” http://msdn.microsoft.com/en-us/library/ms973913.aspx

这些元素提供了否则将存在于注册表中的信息。comInterfaceExternalProxyStub元素提供了足够的信息以进行类型库封送处理,并且适用于从IDispatch派生的COM接口(其中包括所有自动化接口)。在这些情况下,ole32.dll提供了所使用的外部代理存根(即,外部到程序集文件的代理存根)。如果您的COM组件仅实现调度或双重接口,则应使用此元素。


1
IExampleClassEvents 是双重接口还是分发接口?VB6 只能接收源分发接口。请注意,双重接口和分发接口有不同的代理/存根,具有不同的 CLSID:PSOAInterface = {00020424-0000-0000-C000-000000000046} 用于双重接口,而 PSDispatch = {00020420-0000-0000-C000-000000000046} 用于分发接口。您的示例正在使用第二个,因此 VB6 只能使用后期绑定的 IDispatch 调用以使编组在应用程序中工作。(最好使用 PSOAInterface 用于 IExampleClass - wqw
接口是IDispatch(就像我问题中的示例一样)。Regasm还在注册表中使用了这个CLSID。这个接口很久以前就制作好了,我不记得为什么选择了IDispatch。也许是InteropForms Toolkit 就这么做的,我当时以此为例。我的接口非常笨重,因此性能不是真正的问题,使用双重/iunknown还有其他好处吗? - Herman
3
不,你已经通过marshaller获得了足够的开销 :-)) 互联网上充满了代理/存根示例,用于regfree marshaling(包括CLSIDs),但没有清晰的描述哪个何时使用。很明显有两个内置代理,GUID之间也有微妙的差异(只有一个数字)。这些代理不兼容,也不能互换,即你不能在dispinterfaces上使用PSOAInterface -- 最终会出现运行时错误。我在此发布此信息以备后人参考。 - wqw
是的,说得好。当我查看CLSID指向哪个接口时,我不得不再看一次。我的建议是使用regasm /tlb命令,然后从注册表中获取值。 - Herman
1
@herman:iDispatch是COM的标准双接口(即,iUnknown和iDispatch)。VB6 COM需要一个双接口,因为如果只使用iUnknown,则无法支持迟绑定,而VB6不想处理这个问题。所以,你可能是对的;InteropForms Toolkit做到了这一点。更多关于iDispatch的信息请参见此处:http://msdn.microsoft.com/en-us/library/windows/desktop/ms221608(v=vs.85).aspx - BobRodes

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接