我正在尝试用C++构建一个COM库,使用C#项目进行测试.某些方法需要将字符串返回给调用者.从C#调用这些方法后,我得到:"访问违规读取位置..."
这是我的testproject的C++代码(除了VS 2010 ATL生成的所有内容)
//COMTest.idl
[id(1)] HRESULT Test([out,retval] BSTR* ret);
//Program2.h
STDMETHOD(Test)(BSTR* ret);
//Program2.cpp
STDMETHODIMP CProgram2::Test(BSTR* ret)
{
BSTR tmp = (BSTR)CoTaskMemAlloc(sizeof(wchar_t) * 2);
tmp[0] = L'H';
tmp[1] = L'\0';
*ret = (BSTR)tmp;
return S_OK;
}
Run Code Online (Sandbox Code Playgroud)
在C#中,我刚从COM-Tab引用了DLL,关闭了"嵌入互操作类型",因为它导致了错误,并运行了这个:
static void Main(string[] args)
{
COMTestLib.Program2Class instance = new COMTestLib.Program2Class();
string tmp = instance.Test(); //Where the error occurs
Console.WriteLine(tmp); //This is not reached
Console.Read();
}
Run Code Online (Sandbox Code Playgroud)
离开测试方法后发生错误.我在C#项目中调试了C++代码,并将值放在正确的位置.如果我尝试返回0(在C#中为null),即使我仍然像示例中那样分配内存,我也不会收到错误.
我无法理解访问冲突所抱怨的地址.它既不是我分配的地址,也不是该方法中使用的任何其他地址.对我来说似乎很奇怪的是CoTaskMemAlloc-Function总是返回第一个字节设置为零(0x00XXXXXX)的地址,但这可能只是一个COM事物.
我没有想法,我无法在任何地方找到关于此的更多信息(除了基本的COM教程).有人可以帮忙吗?
BSTR 需要额外的内存(以跟踪字符串 len),因此必须使用SysAllocString () 函数来分配 BSTR(或使用“智能”BSTR类之一)。
所以你的原始代码应该是这样的:
//Program2.cpp
STDMETHODIMP CProgram2::Test(BSTR* ret)
{
*ret = SysAllocString(L"H");
return S_OK;
}
Run Code Online (Sandbox Code Playgroud)
关于 BSTR 的好读物:http://blogs.msdn.com/b/ericlippert/archive/2003/09/12/52976.aspx