将C#结构元素编组为C ++ VARIANT

时间:2013-10-31 15:06:36

标签: c# c++ com marshalling

首先,我对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);
    }

在上面的例子中,struct被编组为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-struct中发送的数据(或者在内存中的任何其他位置)

我如何处理VT_VARIANT?

有关如何获取struct-data的任何想法?

更新

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;

2 个答案:

答案 0 :(得分:0)

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

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

答案 1 :(得分: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);
...
}

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