在第三方DLL中调用方法

时间:2013-02-25 08:38:51

标签: c# qt pinvoke marshalling intptr

我正在使用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等的各种组合,但没有发现实际工作的任何内容。

任何帮助都非常感激。

同样感谢网站上的所有贡献者 - 这是一个非常宝贵的资源!

1 个答案:

答案 0 :(得分:3)

您不可能希望使用P / invoke调用这样的C ++库。你只是得到了错误的工具。

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