首先,我对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;
答案 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);
...
}
如果有人知道更好的解决方案,请随意添加。