use*_*370 5 c++ boost boost-phoenix boost-proto
在Boost Phoenix文章中,"转换表达式树",这里是一组自定义invert_actions类的特化,用于反转二进制算术表达式.例如a+b变成a-b; a*b变得a/b; 反之亦然.
这涉及表达式树的递归遍历 - 但是,当遇到涉及未明确处理的运算符的表达式时,此遍历将停止.例如,_1+_2-_3将成为_1-_2+_3,但_1+_1&_2将保持原样(没有处理程序&).let(_a = 1, _b = 2) [ _a+_b ]也将保持不变.
我曾经认为这是本文所预期的,但看看最后列出的测试,我看到if_(_1 * _4)[_2 - _3]预计会发生变化; 使用提供的代码(这里),我发现它没有.
那么如何定义一个通用的Boost Phoenix表达式树变换,它适用于所有一组明确列出的(n-ary)运算符; 让其他人保持不变?
某些代码可能很有用.我想要输出以下C++ 11代码(auto)0,而不是2; 没有显式处理&或任何其他运算符/语句.
#include <iostream>
#include <boost/phoenix.hpp>
#include <boost/proto/proto.hpp>
using namespace boost;
using namespace proto;
using namespace phoenix;
using namespace arg_names;
struct invrt {
template <typename Rule> struct when : proto::_ {};
};
template <>
struct invrt::when<rule::plus>
: proto::call<
proto::functional::make_expr<proto::tag::minus>(
evaluator(_left, _context), evaluator(_right, _context)
)
>
{};
int main(int argc, char *argv[])
{
auto f = phoenix::eval( _1+_1&_2 , make_context(make_env(), invrt()) );
std::cout << f(1,2) << std::endl; // Alas 2 instead of 0
return 0;
}
Run Code Online (Sandbox Code Playgroud)
这是使用直接 Proto 的方法:
#include <iostream>
#include <boost/phoenix.hpp>
#include <boost/proto/proto.hpp>
namespace proto = boost::proto;
using namespace boost::phoenix;
using namespace arg_names;
struct invrt:
proto::or_<
proto::when<
// Turn plus nodes into minus
proto::plus<proto::_, proto::_>,
proto::functional::make_expr<proto::tag::minus>(
invrt(proto::_left), invrt(proto::_right)
)
>,
proto::otherwise<
// This recurses on children, transforming them with invrt
proto::nary_expr<proto::_, proto::vararg<invrt> >
>
>
{};
int main(int argc, char *argv[])
{
auto f = invrt()(_1+_1&_2);
proto::display_expr(f);
std::cout << f(1,2) << std::endl;
return 0;
}
Run Code Online (Sandbox Code Playgroud)
Phoenix 在 Proto 之上分层了一堆东西。我不知道其语义pheonix::eval或为什么您尝试的方法不起作用。也许了解凤凰城的人会插话。
====编辑====
我用凤凰城的例子解决了这个问题。对于非加号情况,它不会递归。您的代码应如下所示:
#include <iostream>
#include <boost/phoenix.hpp>
#include <boost/proto/proto.hpp>
using namespace boost;
using namespace proto;
using namespace phoenix;
using namespace arg_names;
struct invrt {
template <typename Rule>
struct when :
// NOTE!!! recursively transform children and reassemble
nary_expr<_, vararg<proto::when<_, evaluator(_, _context)> > >
{};
};
template <>
struct invrt::when<rule::plus> :
proto::call<
proto::functional::make_expr<proto::tag::minus>(
evaluator(_left, _context), evaluator(_right, _context)
)
>
{};
int main()
{
auto f = phoenix::eval( _1+_1&_2 , make_context(make_env(), invrt()) );
display_expr(f);
std::cout << f(1,2) << std::endl; // Prints 0. Huzzah!
}
Run Code Online (Sandbox Code Playgroud)
您是否认为比直接的 Proto 解决方案更简单或更复杂,由您决定。
| 归档时间: |
|
| 查看次数: |
525 次 |
| 最近记录: |