我使用C#和P-Invoke来访问Qt框架(http://qt.digia.com/)中的对象。使用返回简单类型(或void)的函数似乎没有问题,但每当我尝试使用返回对象的函数时,应用程序就会崩溃。
例如,在QtXml4.dll中,有一个方法QXmlInputSource::data(void),它返回QString类型的对象。下面是我的包装器类:
public class QXmlInputSource
{
// PInvoke - class QString QXmlInputSource::data(void)
[DllImport("QtXml4.dll", CharSet = CharSet.Unicode, EntryPoint = "?data@QXmlInputSource@@UBE?AVQString@@XZ",
SetLastError = true, CallingConvention = CallingConvention.ThisCall)]
static extern IntPtr data(ref IntPtr Ptr);
private IntPtr mPtr;
public QXmlInputSource(IntPtr Ptr)
{
mPtr = Ptr;
}
public override string ToString()
{
IntPtr mData = data(ref mPtr);
return "Epic Fail";
}
}下面是一些代码,它们(使用EasyHook)挂钩到提供有效QXmlInputSource对象的函数调用中:
// just use a P-Invoke implementation to get native API access from C# (this step is not necessary for C++.NET)
[DllImport("QtXml4.dll", CharSet = CharSet.Unicode, EntryPoint = "?parse@QXmlSimpleReader@@UAE_NPBVQXmlInputSource@@@Z",
SetLastError = true, CallingConvention = CallingConvention.ThisCall)]
[return: MarshalAs(UnmanagedType.Bool)]
static extern bool XmlParse(IntPtr Reader, IntPtr Source);
// Intercept all calls to parse XML
public bool XmlParse_Intercepted(IntPtr Reader, IntPtr Source)
{
QXmlInputSource XmlSource = new QXmlInputSource(Source);
String s = XmlSource.ToString();
// call original API...
return XmlParse(Reader, Source);
}钩子代码似乎工作得很好。当我调用包装器类中的data()函数时,Qt应用程序崩溃。正如我上面所说的,每当函数调用返回对象而不是简单类型时,基于Qt的应用程序似乎就会崩溃。
我尝试过CallingConventions、返回类型、封送处理等的各种组合,但没有发现任何实际有效的方法。
任何帮助都非常感谢。
还要感谢网站上的所有贡献者-这是一个无价的资源!
发布于 2013-02-25 22:14:27
您不可能希望使用P/invoke调用这样的C++库。您只是使用了错误的工具来完成工作。
您需要做的是使用C++/CLI混合模式层来完成这项工作。这不仅会有实际工作的明显好处,而且也会容易得多。编写调用本机Qt DLL的C++代码。然后使用托管类将该代码公开给C#。最后,您可以简单地从C#代码中添加对C++/CLI库的引用,这一切都很好。
https://stackoverflow.com/questions/15062983
复制相似问题