1

首先,我对 COM 技术不是很熟悉,这是我第一次使用它,所以请多多包涵。我正在尝试从 C# 调用 COM 对象函数。

这是idl文件中的接口:

[id(6), helpstring("vConnectInfo=ConnectInfoType")] 
HRESULT ConnectTarget([in,out] VARIANT* vConnectInfo);

这是我运行 tlbimp 后得到的互操作接口:

    void ConnectTarget(ref object vConnectInfo);

目标函数的 COM 对象中的 c++ 代码:

STDMETHODIMP PCommunication::ConnectTarget(VARIANT* vConnectInfo)
{
    if (!((vConnectInfo->vt & VT_ARRAY) && (vConnectInfo->vt & VT_BYREF)))
    {
        return E_INVALIDARG;
    }

    ConnectInfoType *pConnectInfo = (ConnectInfoType *)((*vConnectInfo->pparray)->pvData);
...
}

这个 COM 对象在另一个进程中运行,它不在 dll 中。

我可以补充一点,COM 对象也被另一个用 C++ 编写的程序使用。在这种情况下没有问题,因为在 C++ 中创建了一个 VARIANT,并将 pparray->pvData 设置为 connInfo 数据结构,然后使用 VARIANT 作为参数调用 COM 对象。

据我了解,在 C# 中,我的结构应该自动编组为 VARIANT。

这是我一直在使用的两种方法(或者实际上我已经尝试了更多......)从 C# 调用此方法:

    private void method1_Click(object sender, EventArgs e)
    {
        pcom.PCom PCom = new pcom.PCom();
        pcom.IGeneralManagementServices mgmt = (pcom.IGeneralManagementServices)PCom;
        m_ci = new ConnectInfoType();
        fillConnInfo(ref m_ci);
        mgmt.ConnectTarget(m_ci);
    }

在上述情况下,结构被编组为 VT_UNKNOWN。这是一个简单的情况,如果参数不是结构(例如,适用于 int),则可以使用。

    private void method4_Click(object sender, EventArgs e)
    {
        ConnectInfoType ci = new ConnectInfoType();
        fillConnInfo(ref ci);

        pcom PCom = new pcom.PCom();
        pcom.IGeneralManagementServices mgmt = (pcom.IGeneralManagementServices)PCom;

        ParameterModifier[] pms = new ParameterModifier[1];
        ParameterModifier pm = new ParameterModifier(1);
        pm[0] = true;
        pms[0] = pm;
        object[] param = new object[1];
        param[0] = ci;
        object[] args = new object[1];
        args[0] = param;
        mgmt.GetType().InvokeMember("ConnectTarget", BindingFlags.InvokeMethod, null, mgmt, args, pms, null, null);
    }

在这种情况下,它被编组为 VT_ARRAY | VT_BYREF | VT_VARIANT。问题是,在调试“目标函数”ConnectTarget 时,我无法在 SAFEARRAY 结构(或内存中的任何其他位置)中找到我发送的数据

我该如何处理 VT_VARIANT?

关于如何获取我的结构数据的任何想法?

更新:

ConnectInfoType 结构:

    [StructLayout(LayoutKind.Sequential, CharSet = CharSet.Ansi)]
    public class ConnectInfoType
    {
        public short network;
        public short nodeNumber;
        [MarshalAs(UnmanagedType.ByValTStr, SizeConst = 51)]
        public string connTargPassWord;
        [MarshalAs(UnmanagedType.ByValTStr, SizeConst = 8)]
        public string sConnectId;
        [MarshalAs(UnmanagedType.ByValTStr, SizeConst = 16)]
        public string sConnectPassword;
        public EnuConnectType eConnectType;
        public int hConnectHandle;
        [MarshalAs(UnmanagedType.ByValTStr, SizeConst = 8)]
        public string sAccessPassword;
    };

以及 c++ 中的相应结构:

typedef struct ConnectInfoType
{
    short network;
    short nodeNumber;
    char connTargPassWord[51];
    char sConnectId[8];
    char sConnectPassword[16];
    EnuConnectType eConnectType;
    int hConnectHandle;
    char sAccessPassword[8];
} ConnectInfoType;
4

2 回答 2

0

我找到了一个可接受的解决方案/解决方法。我将结构转换为字节数组,然后将接收到的数据转换为 ConnectInfoType 结构。

    // C#
    private void method3_Click(object sender, EventArgs e)
    {
        pcom.PCom PCom = new pcom.PCom();
        pcom.IGeneralManagementServices mgmt = (pcom.IGeneralManagementServices)PCom;
        ConnectInfoType ci = new ConnectInfoType();
        fillConnInfo(ref ci);

        IntPtr ptr = Marshal.AllocHGlobal(Marshal.SizeOf(ci));
        Marshal.StructureToPtr(ci, ptr, false);
        byte[] arr = new byte[Marshal.SizeOf(ci)];
        Marshal.Copy(ptr, arr, 0, Marshal.SizeOf(ci));

        mgmt.ConnectTarget(arr);
    }

这将被编组,VT_ARRAY |VT_UI1因此我将 c++ 代码更改为:

// C++
STDMETHODIMP PCommunication::ConnectTarget(VARIANT* vConnectInfo)
{
    if (!((vConnectInfo->vt & VT_ARRAY) && (vConnectInfo->vt & VT_UI1)))
    {
        return E_INVALIDARG;
    }

    ConnectInfoType *pConnectInfo = (ConnectInfoType *)(vConnectInfo->parray->pvData);
...
}

如果有人知道更好的解决方案,请随时添加。

于 2013-11-05T15:50:06.627 回答
0

使用此类http://msdn.microsoft.com/en-us/library/system.runtime.interopservices.variantwrapper%28v=vs.110%29.aspx对其进行包装,然后调用所需的方法。

您可以发布类型“ConnectInfoType”的 c++ 源代码吗?

于 2013-11-05T12:09:10.233 回答