参考文章:使用python创建生成动态链接库dll
C/C++:Windows编程—调用DLL程序的2种方法
- 编写cython文件run.pyx:
cdef public str_add(str1,str2):
return int(str1) + int(str2)
在run.pyx目录下运行命令:
cython run.pyx
生成run.c和run.h。
cython处理后的文件可以直接在C程序中调用了:
#include <stdio.h>
#include <python.h>
#include "run.h"
int main()
{
Py_Initialize();
int sum = str_add("10", "20");
printf("sum = %d\n", sum);
Py_Finalize();
return 0;
}
- 创建VS dll项目,名为py_dll,将run.c和run.h添加到项目中。
修改项目配置。我选择的解决方案配置为Release | x64,添加了配置好的属性表,将python的include和libs等加入到VC++目录和C/C++常规中。属性 - C/C++ - 预编译头 修改为 不使用预编译头。
修改项目自动生成的dllmain.cpp:
#include <Python.h>
#include <Windows.h>
#include "run.h"
extern "C"
{
__declspec(dllexport) int __stdcall _str_add(const char* a, const char* b) //声明导出函数,类,对象等供外面使用
{
return str_add(a, b);
}
}
BOOL WINAPI DllMain(HINSTANCE hinstDLL, DWORD fdwReason, LPVOID lpReserved)
{
switch (fdwReason)
{
case DLL_PROCESS_ATTACH:
Py_SetPath(L"C:\\Program Files\\Python39\\Lib"); //这个要根据自己电脑python安装的位置来
Py_Initialize();
//dll初始化的时候调用,这是python3的写法,python2改成,initrun()。参见生成的run.h
//PyInit_run();
//这里VS运行的时候报错,建议使用下面这一行:
PyImport_AppendInittab("run", PyInit_run);
break;
case DLL_PROCESS_DETACH:
Py_Finalize();
break;
}
return TRUE;
}
注意:该文件定义了导出函数_str_add,之后在调用的时候不要误用str_add。
-
现在dll项目下的文件有:run.c,run.h, dllmain.cpp。点击运行本地Windows调试器,或者在解决方案“py_dll”右键生成解决方案。
之后在项目根目录下的 x64/Release 可以找到py_dll.dll和py_dll.lib,即为我们要用的链接库。 -
新建一个C/C++项目 use_dll。同样进行配置,将py_dll.dll和py_dll.lib添加到资源文件中。
新建一个cpp文件test.cpp,实现静态调用py_dll.lib、动态调用py_dll.dll:
#include <stdio.h>
#include <stdlib.h>
#include <Windows.h>
#pragma comment(lib,"py_dll.lib")
extern "C" __declspec(dllexport) int __stdcall _str_add(const char* a, const char* b); //声明导出函数,类,对象等供外面使用
// 静态调用DLL库
void StaticUse()
{
int sum = _str_add("10", "20");
printf("静态调用,sum = %d\n", sum);
}
// 动态调用DLL库
void DynamicUse()
{
HMODULE module = LoadLibrary(L"py_dll.dll");
if (module == NULL)
{
printf("加载py_dll.dll动态库失败\n");
return;
}
typedef int(*Func)(const char*, const char*); // 定义函数指针类型
Func _str_add;
_str_add = (Func)GetProcAddress(module, "_str_add");
int sum = _str_add("100", "200");
printf("动态调用,sum = %d\n", sum);
}
int main(char argc, char* argv[])
{
StaticUse();
DynamicUse();
system("pause");
return 0;
}
输出结果: