我希望这是一个简单的 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
。如果我要使用另一个参数组合,我相信我必须使用另一个函数名称。在这种情况下,重载似乎失败了。这很好,但开发人员的警告就更好了。