将一个C ++ DLL加载到matlab中,该matlab调用另一个DLL中的函数

时间:2014-11-14 10:35:30

标签: c++ dll call

出于学习目的,我正在尝试将DLL加载到MATLAB中,该MATLAB调用另一个DLL中定义的函数。我是所有这一切的新手,还没有弄清楚我将如何做到这一点,也没有设法找到任何相关的资源。

我在C ++中编写了一个小DLL,其内容如下:

//example_dll.h
#ifndef EXAMPLE_DLL_H
#define EXAMPLE_DLL_H

#ifdef __cplusplus
extern "C" {
#endif

#ifdef BUILDING_EXAMPLE_DLL
#define EXAMPLE_DLL __declspec(dllexport)
#else
#define EXAMPLE_DLL __declspec(dllimport)
#endif

int EXAMPLE_DLL Double(int x);

#ifdef __cplusplus
}
#endif

#endif  // EXAMPLE_DLL_H

和源文件:

//example_dll.cpp
#include <stdio.h>
#include "example_dll.h"

int Double(int x)
{
        return 2 * x;
}

我使用MinGW w64构建并使用loadlibrary('example_dll')加载到matlab中,没有任何问题。

我现在想要定义函数

int Double(int x)
{
        return 2 * x;
}

在另一个DLL中,(我们称之为DLL2)并从我的example_dll中调用该函数。 最简单的方法是什么?

我希望得到一个简短的示例代码(最好是运行时动态链接,不使用模块定义(.def)文件)或链接到互联网上的相关资源。 谢谢!

解决方案简单示例:

我想我得到了解决方案。无论如何它似乎都在起作用。

我创建了一个名为interface_DLL的DLL,我将其加载到MATLAB中,然后在example_dll中调用了我的函数

这里是:

//interface_dll.h
#ifndef INTERFACE_DLL_H
#define INTERFACE_DLL_H

#ifdef __cplusplus
extern "C" {
#endif

#ifdef BUILDING_INTERFACE_DLL
#define INTERFACE_DLL __declspec(dllexport)
#else
#define INTERFACE_DLL __declspec(dllimport)
#endif


int INTERFACE_DLL Quadruple(int x);

#ifdef __cplusplus
}
#endif

#endif  // INTERFACE_DLL_H

和源文件:

//interface_dll.cpp
#include <windows.h>
#include <stdio.h>
#include "interface_dll.h"
#include "example_dll.h"

int Quadruple(int x)
{
    /* get handle to dll */ 
    HINSTANCE hGetProcIDDLL = LoadLibrary("C:\\Users\\uidr0605\\Documents\\ExampleDLL\\example_dll.dll"); 

    /* get pointer to the function in the dll*/
    FARPROC lpfnGetProcessID = GetProcAddress(HMODULE (hGetProcIDDLL),"Double");

    /*
    Define the Function in the DLL for reuse. This is just prototyping the dll's function.
    A mock of it. Use "stdcall" for maximum compatibility.
    */
    typedef int (__stdcall * pICFUNC)(int);

    pICFUNC Double;
    Double = pICFUNC(lpfnGetProcessID);

    /* The actual call to the function contained in the dll */
    int intMyReturnVal = Double(x);
    intMyReturnVal = Double(intMyReturnVal);

    /* Release the Dll */
    FreeLibrary(hGetProcIDDLL);

    /* The return val from the dll */
    return intMyReturnVal; 
}

我从MATLAB加载它如下:

%loadDLL.m
path = 'C:\Path\to\DLL\';
addpath(path);
loadlibrary('interface_dll')
i = 2;
x = calllib('interface_dll', 'Quadruple', i)

我正在完成此过程的原因是the MATLAB shared library interface supports C library routines only and not C++ classes.

我对解决方法的想法是使用中间DLL作为MATLAB和我打算访问的类的DLL之间的接口。 有没有更好的方法呢?

更多问题:

有人可以解释这里适用的typedef int (__stdcall * pICFUNC)(int);行的重要性吗? 如果我想在example_dll的类中调用函数,我需要添加什么或者我需要做些什么更改?

编辑:我将以下代码添加到example_dll头文件中:

class EXAMPLE_DLL MyClass
{
public: 
        int add2(int);
};


#ifdef __cplusplus
extern "C" {
#endif


MyClass EXAMPLE_DLL *createInstance(){
        return new MyClass();
}

void EXAMPLE_DLL destroyInstance(MyClass *ptrMyClass){
        delete ptrMyClass;
}

#ifdef __cplusplus
}
#endif

1 个答案:

答案 0 :(得分:2)

进一步的问题1

以下定义

typedef int (__stdcall * pICFUNC)(int);

定义了一个新类型pICFUNC,它是一个指向函数的指针,该函数接受一个int并返回一个int。此外,必须根据__stdcall调用约定调用该函数,该约定指定必须如何传递参数以及如何检索返回值。 这个link解释了带有函数指针的typedef。另请参阅以下部分,使用带有类型强制转换的typedef ,因为在线

Double = pICFUNC(lpfnGetProcessID);

pICFUNC用来代替。

进一步的问题2

以下是一个非常简单的例子来提出一个想法。如果你在 example_dll 中有一个名为 MyClass 的类,它有一个方法

int add(int num);

您可以实现以下功能:

MyClass *createInstance(){
    return new MyClass();
}

void destroyInstance(MyClass *ptrMyClass){
    delete ptrMyClass;
}

这些功能必须是 extern“C”,您可以使用 GetProcAddress 导入它们。然后,只需要创建一个实例,通过指针调用它的方法并最终销毁它。

编辑:有关实施的一些提示

导入函数以创建实例

FARPROC lpfnCreateInstance = GetProcAddress(HMODULE (hGetProcIDDLL), "createInstance");

为函数声明一个正确的指针类型(返回一个MyClass *,没有参数)

typedef MyClass* (__stdcall * pCREATINST)();

施放lpfnCreateInstance

pCREATINST createInstance;

createInstance = pCREATINST(lpfnCreateInstance)

创建您的实例

MyClass *myInstance = creatInstance();

然后你不需要一个add的包装器,你可以从你的指针调用它。

int res = myInstance->add(123);

你应该对destroyInstance做同样的事情,小心这些类型 请注意,我无法测试此代码,但它应该是正确的方法。