出于学习目的,我正在尝试将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
答案 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做同样的事情,小心这些类型 请注意,我无法测试此代码,但它应该是正确的方法。