SWIG Python C++ 结构作为输入/输出参数

dev*_*kim 3 c++ python swig struct

老实说,我在这个网站上阅读并重新阅读了很多有关结构主题的帖子。但我需要你的帮助。

我有 C 风格的结构

   struct Time
   {
      uint16_t  year; //    year with four digits like 2016
      uint8_t   month; // 1 .. 12
      uint8_t   day; // 1 .. 31
      uint8_t   hour; // 0 .. 23, 24 hour representation
      uint8_t   minute; // 0 .. 59
      uint8_t   second; // 0 .. 59
   };
Run Code Online (Sandbox Code Playgroud)

以及带有成员函数的类,其实现在DLL中。

class DeviceInterface {
virtual uint32_t getTime(Time&  time) = 0;
};
Run Code Online (Sandbox Code Playgroud)

其中uint32_t值是状态代码。

这是自动生成的 SWIG C++ 代码:

SWIGINTERN PyObject *_wrap_getTime(PyObject *SWIGUNUSEDPARM(self), PyObject *args) {
  PyObject *resultobj = 0;
  DeviceInterface *arg1 = (DeviceInterface *) 0 ;
  Time *arg2 = 0 ;
  void *argp1 = 0 ;
  int res1 = 0 ;
  void *argp2 = 0 ;
  int res2 = 0 ;
  PyObject *swig_obj[2] ;
  uint32_t result;
  
  if (!SWIG_Python_UnpackTuple(args, "getTime", 2, 2, swig_obj)) SWIG_fail;
  res1 = SWIG_ConvertPtr(swig_obj[0], &argp1,SWIGTYPE_p_DeviceInterface, 0 |  0 );
  if (!SWIG_IsOK(res1)) {
    SWIG_exception_fail(SWIG_ArgError(res1), "in method '" "getTime" "', argument " "1"" of type '" "DeviceInterface *""'"); 
  }
  arg1 = reinterpret_cast< DeviceInterface * >(argp1);
  res2 = SWIG_ConvertPtr(swig_obj[1], &argp2, SWIGTYPE_p_Time,  0 );
  if (!SWIG_IsOK(res2)) {
    SWIG_exception_fail(SWIG_ArgError(res2), "in method '" "getTime" "', argument " "2"" of type '" "Time &""'"); 
  }
  if (!argp2) {
    SWIG_exception_fail(SWIG_ValueError, "invalid null reference " "in method '" "getTime" "', argument " "2"" of type '" "Time &""'"); 
  }
  arg2 = reinterpret_cast< Time * >(argp2);
  result = (uint32_t)(arg1)->getTime(*arg2);
  resultobj = SWIG_From_unsigned_SS_int(static_cast< unsigned int >(result));
  return resultobj;
fail:
  return NULL;
}
Run Code Online (Sandbox Code Playgroud)

从上面的代码中我看不到以某种方式返回了什么时间值,即arg2变量。那么我需要在 SWIG 接口文件中编写什么才能获取状态代码和时间

Mar*_*nen 5

您可以编写类型映射来附加Time输出参数。SWIG 生成结构的代理,可以根据需要通过 生成SWIG_NewPointerObj()。完整示例如下:

DeviceInterface.h(最小实现)

#include <stdint.h>

struct Time
{
    uint16_t year;   // year with four digits like 2016
    uint8_t  month;  // 1 .. 12
    uint8_t  day;    // 1 .. 31
    uint8_t  hour;   // 0 .. 23, 24 hour representation
    uint8_t  minute; // 0 .. 59
    uint8_t  second; // 0 .. 59
};

class DeviceInterface {
public:
    virtual uint32_t getTime(Time& time) {
        time.year = 2021;
        time.month = 12;
        time.day = 30;
        time.hour = 9;
        time.minute = 47;
        time.second = 30;
        return 0;
    }
};
Run Code Online (Sandbox Code Playgroud)

测试.i

%module test

%{
#include "DeviceInterface.h"
%}

%include <stdint.i>

// Do not require an input parameter for Time.
// Instead, SWIG will allocate one.
%typemap(in,numinputs=0) Time& %{
    $1 = new Time;
%}

// After calling the function, process Time as an output.
// Convert the allocated pointer to a SWIG Python wrapper
// and pass ownership to Python.  Python will free the object
// when it goes out of scope.
%typemap(argout) Time& (PyObject* tmp) %{
    // Convert C pointer to SWIG Python wrapper
    tmp = SWIG_NewPointerObj($1, $1_descriptor, SWIG_POINTER_OWN);
    // Append to existing uint32_t return value
    $result = SWIG_Python_AppendOutput($result, tmp);
%}

%include "DeviceInterface.h"
Run Code Online (Sandbox Code Playgroud)

演示:

>>> import test
>>> d=test.DeviceInterface()
>>> r=d.getTime()
>>> r
[0, <test.Time; proxy of <Swig Object of type 'Time *' at 0x000001ED4F866090> >]
>>> r[1].year
2021
>>> r[1].month
12
Run Code Online (Sandbox Code Playgroud)

如果您想自定义 SWIG 包装器的显示以使其Time更易于阅读,您可以使用以下内容扩展 Time 对象:

>>> import test
>>> d=test.DeviceInterface()
>>> r=d.getTime()
>>> r
[0, <test.Time; proxy of <Swig Object of type 'Time *' at 0x000001ED4F866090> >]
>>> r[1].year
2021
>>> r[1].month
12
Run Code Online (Sandbox Code Playgroud)

演示:

%module test

%{
// Added to support the __repr__ implementation
#include <string>
#include <sstream>

#include "DeviceInterface.h"
%}

%include <stdint.i>
%include <std_string.i> // SWIG support for std::string.

%typemap(in,numinputs=0) Time& %{
    $1 = new Time;
%}

%typemap(argout) Time& (PyObject* tmp) %{
    tmp = SWIG_NewPointerObj($1, $1_descriptor, SWIG_POINTER_OWN);
    $result = SWIG_Python_AppendOutput($result, tmp);
%}

// Extend time to suport Python's __repr__.
// It must return a string representing how to display the object in Python.
%extend Time {
    std::string __repr__()
    {
        std::ostringstream ss;
        ss << "Time(year=" << $self->year << ", month=" << (unsigned)$self->month
           << ", day=" << (unsigned)$self->day << ", hour=" << (unsigned)$self->hour
           << ", minute=" << (unsigned)$self->minute << ", second=" << (unsigned)$self->second << ")";
        return ss.str();
    }
}

%include "DeviceInterface.h"
Run Code Online (Sandbox Code Playgroud)