diff options
Diffstat (limited to 'src/boost/libs/yap/example/calc3.cpp')
-rw-r--r-- | src/boost/libs/yap/example/calc3.cpp | 100 |
1 files changed, 100 insertions, 0 deletions
diff --git a/src/boost/libs/yap/example/calc3.cpp b/src/boost/libs/yap/example/calc3.cpp new file mode 100644 index 00000000..7f0f55b7 --- /dev/null +++ b/src/boost/libs/yap/example/calc3.cpp @@ -0,0 +1,100 @@ +// Copyright (C) 2016-2018 T. Zachary Laine +// +// Distributed under the Boost Software License, Version 1.0. (See +// accompanying file LICENSE_1_0.txt or copy at +// http://www.boost.org/LICENSE_1_0.txt) +//[ calc3 +#include <boost/yap/expression.hpp> + +#include <boost/hana/maximum.hpp> + +#include <iostream> + + +// Look! A transform! This one transforms the expression tree into the arity +// of the expression, based on its placeholders. +//[ calc3_get_arity_xform +struct get_arity +{ + // Base case 1: Match a placeholder terminal, and return its arity as the + // result. + template <long long I> + boost::hana::llong<I> operator() (boost::yap::expr_tag<boost::yap::expr_kind::terminal>, + boost::yap::placeholder<I>) + { return boost::hana::llong_c<I>; } + + // Base case 2: Match any other terminal. Return 0; non-placeholders do + // not contribute to arity. + template <typename T> + auto operator() (boost::yap::expr_tag<boost::yap::expr_kind::terminal>, T &&) + { + using namespace boost::hana::literals; + return 0_c; + } + + // Recursive case: Match any expression not covered above, and return the + // maximum of its children's arities. + template <boost::yap::expr_kind Kind, typename... Arg> + auto operator() (boost::yap::expr_tag<Kind>, Arg &&... arg) + { + return boost::hana::maximum( + boost::hana::make_tuple( + boost::yap::transform( + boost::yap::as_expr(std::forward<Arg>(arg)), + get_arity{} + )... + ) + ); + } +}; +//] + +int main () +{ + using namespace boost::yap::literals; + + // These lambdas wrap our expressions as callables, and allow us to check + // the arity of each as we call it. + + auto expr_1 = 1_p + 2.0; + + auto expr_1_fn = [expr_1](auto &&... args) { + auto const arity = boost::yap::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::yap::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::yap::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; +} +//] |