2

我正在使用 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、返回类型、Marshaling 等的各种组合,但没有发现任何真正有效的方法。

非常感谢任何帮助。

还要感谢网站上的所有贡献者——这是一个非常宝贵的资源!

4

1 回答 1

3

您不可能希望使用 P/invoke 调用这样的 C++ 库。您只是使用了错误的工具来完成这项工作。

您需要做的是使用 C++/CLI 混合模式层来完成这项工作。这不仅对实际工作有明显的好处,而且也容易得多。编写调用本机 Qt DLL 的 C++ 代码。然后使用托管类将该代码公开给您的 C#。最后,您可以简单地从您的 C# 代码中添加对 C++/CLI 库的引用,这一切都很好。

于 2013-02-25T14:14:27.027 回答