7

我正在尝试从使用 SWIG for Python 包装的 C 函数输出值数组。我尝试做的方式是使用以下类型映射。

伪代码:

int oldmain() {
float *output = {0,1};
return output;
}

类型图:

%typemap(out) float* { 
   int i; 
  $result = PyList_New($1_dim0); 
   for (i = 0; i < $1_dim0; i++) { 
 PyObject *o = PyFloat_FromDouble((double) $1[i]); 
 PyList_SetItem($result,i,o); 
 } 
} 

我的代码编译得很好,但是当我运行访问这个函数时它挂起(没有更多的方法来调试它)。

关于我哪里出错的任何建议?

谢谢。

4

2 回答 2

7

允许长度变化的最简单方法是添加另一个输出参数,该参数也告诉您数组的大小:

%module test

%include <stdint.i>

%typemap(in,numinputs=0,noblock=1) size_t *len  {
  size_t templen;
  $1 = &templen;
}

%typemap(out) float* oldmain {
  int i;
  $result = PyList_New(templen);
  for (i = 0; i < templen; i++) {
    PyObject *o = PyFloat_FromDouble((double)$1[i]);
    PyList_SetItem($result,i,o);
  }
}

%inline %{
float *oldmain(size_t *len) {
  static float output[] = {0.f, 1.f, 2, 3, 4};
  *len = sizeof output/sizeof *output;
  return output;
}
%}

这是从这个答案修改为添加size_t *len可用于在运行时返回数组的长度。类型映射完全隐藏了 Python 包装器的输出,而是使用它%typemap(out)而不是固定大小来控制返回列表的长度。

于 2012-07-27T07:40:27.670 回答
6

这应该可以帮助您:

/* example.c */

float * oldmain() {
    static float output[] = {0.,1.};
    return output;
}

您在这里返回一个指针,而 swig 不知道它的大小。普通的 $1_dim0 是行不通的,所以你必须硬编码或做一些其他的魔法。像这样的东西:

/* example.i */
%module example
%{
 /* Put header files here or function declarations like below */
  extern float * oldmain();
%}

%typemap(out) float* oldmain {
  int i;
  //$1, $1_dim0, $1_dim1
  $result = PyList_New(2);
  for (i = 0; i < 2; i++) {
    PyObject *o = PyFloat_FromDouble((double) $1[i]);
    PyList_SetItem($result,i,o);
  }
}

%include "example.c"

然后在python中你应该得到:

>> import example
>> example.oldmain()
[0.0, 1.0]

添加类型映射时,您可能会发现-debug-tmsearch非常方便,即

swig -python -debug-tmsearch example.i

在为float *oldmain. 此外,如果您只是想访问 c 全局变量数组,您可以使用 typemap forvarout而不是 just来执行相同的技巧out

于 2012-07-27T02:05:08.000 回答