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