#include "expression.hpp" #include #include struct get_arity { template auto operator() (Expr const & expr) { if constexpr (Expr::kind == boost::proto17::expr_kind::placeholder) { return expr.value(); } else if constexpr (Expr::kind == boost::proto17::expr_kind::terminal) { using namespace boost::hana::literals; return 0_c; } else { return boost::hana::maximum( boost::hana::transform( expr.elements, [](auto const & element) { return boost::proto17::transform(element, get_arity{}); } ) ); } } }; int main () { using namespace boost::proto17::literals; auto expr_1 = 1_p + 2.0; auto expr_1_fn = [expr_1](auto &&... args) { auto const arity = boost::proto17::transform(expr_1, get_arity{}); static_assert(arity.value == sizeof...(args), "Called with wrong number of args."); return evaluate(expr_1, args...); }; auto expr_2 = 1_p * 2_p; auto expr_2_fn = [expr_2](auto &&... args) { auto const arity = boost::proto17::transform(expr_2, get_arity{}); static_assert(arity.value == sizeof...(args), "Called with wrong number of args."); return evaluate(expr_2, args...); }; auto expr_3 = (1_p - 2_p) / 2_p; auto expr_3_fn = [expr_3](auto &&... args) { auto const arity = boost::proto17::transform(expr_3, get_arity{}); static_assert(arity.value == sizeof...(args), "Called with wrong number of args."); return evaluate(expr_3, args...); }; // Displays "5" std::cout << expr_1_fn(3.0) << std::endl; // Displays "6" std::cout << expr_2_fn(3.0, 2.0) << std::endl; // Displays "0.5" std::cout << expr_3_fn(3.0, 2.0) << std::endl; // Static-asserts with "Called with wrong number of args." //std::cout << expr_3_fn(3.0) << std::endl; // Static-asserts with "Called with wrong number of args." //std::cout << expr_3_fn(3.0, 2.0, 1.0) << std::endl; return 0; }