I am trying to use c++11 lambdas as accessor functions in boost::python
's add_property
, something along the following (the lambda is not strictly needed in this example, but will be needed for more complicated things happening inside the lambda, such as validation):
#include<boost/python.hpp>
struct A{
A(): a(2){};
int a;
};
BOOST_PYTHON_MODULE(boost_python_lambda)
{
boost::python::class_<A>("A")
// .def_readonly("a",&A::a) // the classical way: works fine
.add_property("a",[](const A& a){return a.a;})
;
}
However, compiling with clang++ (ver. 3.2) and -std=c++11
(the result is the same with g++ 4.7), I get this error:
/usr/include/boost/python/class.hpp:442:66: error: no matching function for call to 'get_signature'
return python::make_function(f, default_call_policies(), detail::get_signature(f, (T*)0));
^~~~~~~~~~~~~~~~~~~~~
/usr/include/boost/python/class.hpp:422:22: note: in instantiation of function template specialization 'boost::python::class_<A,
boost::python::detail::not_specified, boost::python::detail::not_specified,
boost::python::detail::not_specified>::make_fn_impl<A, <lambda at boost_python_lambda.cpp:12:21> >' requested here
return this->make_fn_impl(
^
/usr/include/boost/python/class.hpp:309:40: note: in instantiation of function template specialization 'boost::python::class_<A,
boost::python::detail::not_specified, boost::python::detail::not_specified,
boost::python::detail::not_specified>::make_getter<<lambda at boost_python_lambda.cpp:12:21> >' requested here
base::add_property(name, this->make_getter(fget), docstr);
^
boost_python_lambda.cpp:12:4: note: in instantiation of function template specialization 'boost::python::class_<A,
boost::python::detail::not_specified, boost::python::detail::not_specified,
boost::python::detail::not_specified>::add_property<<lambda at boost_python_lambda.cpp:12:21> >' requested here
.add_property("a",[](const A& a){return a.a;})
^
I tried wrapping the lambda in std::function<int(const A&)>(...)
, but that did not help with the argument deduction. Any idea?
Use the
make_function()
function to create Python callable objects. If Boost.Python cannot deduce the function object signature, then the signature must be explicitly provided as an MPL front-extensible sequence. For example, the lambda[](const A& a) { return a.a; }
returns anint
and acceptsconst A&
, so one could useboost::mpl::vector<int, const A&>()
for the signature.Here is a complete example demonstrating using a pointer-to-data-member, casting non-capturing lambda to a function, and using a lambda/functor:
Interactive usage:
An alternative non-intrusive approach that is based on documented behavior is to write the lambda as a non-member function, then exposing it as the
fget
argument. While not as succinct as lambda, it still allows for additional functionality, such as validation, to occur when accessing the member variable.If you make the function type explicit by creating an std::function, then using the following piece of (C++11) code you can do it
example:
Hopping in here two years later, Boost.Python indeed does not support wrapping function objects. But your lambda does not capture anything. As such, it can be explicitly convertible to a function pointer:
All you need is that
+
.I've encountered the same problem when trying to use a lambda, and based on the solution for
std::function
above, I've added some more template magic, deducing the type of theoperator()
member function of the lambda (or functor):https://gist.github.com/YannickJadoul/013d3adbf7f4e4abb4d5
And then things like this just work, when this header is included:
The only caveat that I know of, ftm, is that you should include this header befóre including
boost/python.hpp
(or any other Boost.Python header you are using that will need theget_signature
function declarations):