python - 在 OR 工具中通过 SWIG 使用 Python 回调

标签 python c++ swig

我希望这是一个简单的 SWIG 问题。我正在使用 Google OR-Tools 优化库。它是一个包装在 SWIG 中的 C++ 库(我对此知之甚少)。我很难让 Python 回调函数正常工作。有一个C++函数

DecisionBuilder* MakePhase(const std::vector<IntVar*>& vars,
                           IndexEvaluator1* var_evaluator,
                           IntValueStrategy val_str);

连同

typedef ResultCallback1<int64, int64> IndexEvaluator1;

相关的 SWIG(我相信)是

  DecisionBuilder* VarEvalValStrPhase(
      const std::vector<IntVar*>& vars,
      ResultCallback1<int64, int64>* var_evaluator,
      operations_research::Solver::IntValueStrategy val_str) {
    return self->MakePhase(vars, var_evaluator, val_str);
  }

在另一个 SWIG 文件中我们有

%{
static int64 PyCallback1Int64Int64(PyObject* pyfunc, int64 i) {
  // () needed to force creation of one-element tuple
  PyObject* pyresult = PyEval_CallFunction(pyfunc, "(l)", static_cast<long>(i));
  int64 result = 0;
  if (!pyresult) {
    PyErr_SetString(PyExc_RuntimeError,
                    "ResultCallback1<int64, int64> invocation failed.");
  } else {
    result = PyInt_AsLong(pyresult);
    Py_DECREF(pyresult);
  }
  return result;
}
%}

%typemap(in) ResultCallback1<int64, int64>* {
  if (!PyCallable_Check($input)) {
    PyErr_SetString(PyExc_TypeError, "Need a callable object!");
    SWIG_fail;
  }
  $1 = NewPermanentCallback(&PyCallback1Int64Int64, $input);
}

在我的 Python 模块中,我定义了一个函数 Run1,如下所示(我认为这里应该有一些类型转换,但我认为这不是 Python 的方式):

def Run1(index1):
    return index1

并设置

selector_callback = Run1
solver = pywrapcp.Solver("graph-coloring")

最后,我打电话

solver.Phase(nodes,
             selector_callback,
             solver.INT_VALUE_DEFAULT)

唉,这就是卡布洛伊的地方,我总是得到以下错误:

  File "C:\dev\Python27\lib\site-packages\ortools-1.3853-py2.7-win-amd64.egg\ortools\constraint_solver\pywrapcp.py", line 457, in Phase
    def Phase(self, *args): return _pywrapcp.Solver_Phase(self, *args)
NotImplementedError: Wrong number or type of arguments for overloaded function 'Solver_Phase'.
  Possible C/C++ prototypes are:
    operations_research::Solver::MakePhase(std::vector< operations_research::IntVar *,std::allocator< operations_research::IntVar * > > const &,operations_research::Solver::IntVarStrategy,operations_research::Solver::IntValueStrategy)
    operations_research::Solver::MakePhase(std::vector< operations_research::IntervalVar *,std::allocator< operations_research::IntervalVar * > > const &,operations_research::Solver::IntervalStrategy)
    operations_research::Solver::MakePhase(std::vector< operations_research::SequenceVar *,std::allocator< operations_research::SequenceVar * > > const &,operations_research::Solver::SequenceStrategy)

困难在于第二个参数中的回调函数;如果我使用内置值之一而不是回调,则操作成功。但是,我确实需要在那里有自己的功能。

我没有在我的模块中导入任何 SWIG 文件。我需要这样做吗?

最佳答案

所以几天后,我找到了答案。如果我打电话

solver.VarEvalValStrPhase(nodes,
             selector_callback,
             solver.INT_VALUE_DEFAULT)

代替了整个手册中引用的标准函数名称 Phase,它起作用了。如果我要使用另一个参数组合,我相信我必须使用另一个函数名。在这种情况下,重载似乎失败了。这很好,但开发人员发出警告会更好。

关于python - 在 OR 工具中通过 SWIG 使用 Python 回调,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/29090752/

相关文章:

python - 将 POST 和 GET 发送到同一函数时出现 flask 错误,

c++ - 如何干净利落地结束 Lua 线程?

lua - Swig 类型转换为派生类?

c++ - 是否可以拆分 SWIG 模块进行编译,但在链接时重新加入它?

Python:使用命令行或导入

python - 访问脚本主模块内定义的python类变量

c++ - 即使有调试编译标志,表达式也是 "optimized out"

python - 为 numpy 创建一个简单的 SWIG 类型映射

python - 在python列表中将数字转换为成绩

c# - Windows 上等效的 UpdateSystemActivity 是什么