Warning: file_get_contents(/data/phpspider/zhask/data//catemap/5/url/2.json): failed to open stream: No such file or directory in /data/phpspider/zhask/libs/function.php on line 167

Warning: Invalid argument supplied for foreach() in /data/phpspider/zhask/libs/tag.function.php on line 1116

Notice: Undefined index: in /data/phpspider/zhask/libs/function.php on line 180

Warning: array_chunk() expects parameter 1 to be array, null given in /data/phpspider/zhask/libs/function.php on line 181
C# 第三方DLL中的调用方法_C#_Qt_Pinvoke_Marshalling_Intptr - Fatal编程技术网

C# 第三方DLL中的调用方法

C# 第三方DLL中的调用方法,c#,qt,pinvoke,marshalling,intptr,C#,Qt,Pinvoke,Marshalling,Intptr,我使用C#和p-Invoke来访问Qt框架()中的对象。使用返回简单类型(或void)的函数似乎没有问题,但每当我尝试使用返回对象的函数时,应用程序就会崩溃 例如,在qtml4.dll中,有一个方法QXmlInputSource::data(void)返回类型为QString的对象。这是我的包装器类: public class QXmlInputSource { // PInvoke - class QString QXmlInputSource::data(void) [Dll

我使用C#和p-Invoke来访问Qt框架()中的对象。使用返回简单类型(或void)的函数似乎没有问题,但每当我尝试使用返回对象的函数时,应用程序就会崩溃

例如,在qtml4.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的应用程序似乎就会崩溃

我尝试了各种调用约定、返回类型、封送处理等组合,但没有发现任何真正有效的组合

非常感谢您的帮助


同时,也要感谢网站上所有的贡献者——这是一个无价的资源

>你不可能希望用p/Unjk调用这样的C++库。你只是用错了这项工作的工具


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

您是否尝试过对入口点进行ommit操作,而只是尝试直接访问库?没有,因为没有EntryPoint属性,我不知道静态extern声明使用什么函数名。请记住,这些是对象的成员函数,而不是静态函数。修饰过的名称(例如“?”?parse@QXml...)是识别正确方法所必需的。因此,您是否考虑过,如果您没有对象(实例),则无法调用方法?我有。我正在使用Qt应用程序中的实例-通过挂接特定调用,可以获得对它的IntPtr引用。“IntPtr Source”是对QXmlInputSource实例的引用,我试图通过调用其上的data()方法来检查其中的XML数据。您是否试图在读取之前返回数据??在调用XmlParse之前,您正在调用
.ToString()
,这是有意的吗?希望总是有的!!:P不过我相信你的话。我肯定一两天后我会带着另一个问题回来……直到你有更多的代表,你所能做的就是接受答案。无论如何,我会给你一些代表性,通过投票你的问题,这是一个好问题。我经常忘记对问题进行表决。