【发布时间】:2017-10-24 12:21:52
【问题描述】:
我正在使用 Swig 为 DLL 文件生成 python 包装器。我要做的是:
- 使用
swig -c++ -python myfile.i生成包装器和加载器文件 - 创建一个新的 DLL 文件,包括 myfile_wrapper.cxx 和 编译为 _myfile.pyd。
- 在 Idle 中加载 Swig 创建的模块 myfile.py 并尝试使用它。
接口文件如下:
%module myfile
/* Make a test with cpointer - needed?? */
%include cpointer.i
%pointer_functions(MyHandle, my_handle_p);
%pointer_class(int, intp);
%{
#define SWIG_FILE_WITH_INIT
#include "MyFile.h"
}%
%include "MyFile.h"
函数看起来像
typedef struct tagMyHandle
{
void* reserved;
} *MyHandle;
int OpenFile(const char *szPath, MyHandle* pFile); // pFile is an out parameter
int GetNumberOfItems(MyHandle hFile, int *pnNrOfItems); // pnNrOfItems is an out parameter
如果我尝试从 Python 中使用它,我必须这样做:
import myfile
handle_p = myfile.new_my_handle_p()
myfile.OpenFile("Path", handle_p)
handle = myfile.my_file_p_value(handle_p)
num_items_p = myfile.new_intp()
myfile.GetNumberOfItems(handle, num_items_p)
num_items = num_items_p.value()
我是否错误地使用了 Swig?感觉调用应该为Python包装的函数是一种很繁琐的方式。
我想做这样的事情:
result, handle = OpenFile("path")
result, items = GetNumberIfItems(handle)
我无法更改 myfile.h 的源代码。
我查看了input/output parameters,但我必须为每种输出类型定义它们吗? MyFile.h 有数百个具有不同输出类型的函数。而且它只支持原始数据类型,但是MyFile.h中的大多数类型都不是原始类型,而是像struct MyHandle一样。
我也看过SWIG function with pointer struct 和http://www.swig.org/Doc3.0/Python.html#Python_nn18,但没有任何好的解决方案。
更新 1 经过多方帮助,大部分问题我都解决了,但还有一些我不明白的地方。
问题一:
// For the out parameter, shouldn't be needed?
%typemap(in,numinputs=0) MyHandle* pOutParam (MyHandle h) %{
$1 = &h;
%}
// For all others
%typemap(in,numinputs=0) MyHandle* (MyHandle h) %{
$1 = &h;
%}
// For the return type
%typemap(argout) MyHandle* pOutParam (PyObject* o) %{
o = PyLong_FromVoidPtr(*$1);
$result = SWIG_Python_AppendOutput($result,o);
%}
%typemap(in) MyHandle %{
$1 = reinterpret_cast<MyHandle>(PyLong_AsVoidPtr($input));
%}
和代码
int OpenFile(const char *szPath, MyHandle* pOutParam);
int DoSomething(MyHandle* pInParam);
OpenFile 像魅力一样工作,但 DoSomething 仍然尝试返回 MyHandle 而不是将其作为 in 参数,我不明白为什么。 %typemap(argout) MyHandle* 仅为pOutParam 定义。
问题 2: 我不明白如何为类似的东西制作类型映射
int GetFileName(char *szPathBuffer, int iLength);
如何创建一个 char 缓冲区并将其发送进去,就像 I C:
char szBuffer[MAX_PATH]; GetFileName(szBuffer, MAX_PATH);
也许与cstring_bounded_output 一起做一些事情,或者我应该做一些类似的事情
%typemap(in) (char*, int) {
$2 = PyString_Size($input);
$1 = (char*) malloc($2 * sizeof(char*));
}
但它在哪里被释放?
问题 3: 枚举值的正确映射是什么。如果我有
typedef enum tagMyEnum {
MyTrue = 1,
MyFalse = 0 } MyEnum;
和功能
int IsCorrect(MyEnum* pOutValue);
@马克托隆宁: 感谢大家的帮助!对此,我真的非常感激!我学到了很多关于 Swig 的新东西!
【问题讨论】:
-
是和不是。这只是使用 SWIG 的一种方式。您可以创建更高级的类型映射,它允许更简单的界面,例如参见如何使用
numpy.i包装 NumPy。如果您不熟悉编写高级类型映射,我可以推荐%extend功能,其中扩展是用目标语言制作的