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]
が予想されます。提供されたコード(ここ)では、そうではないことがわかりました。
次に、明示的にリストされた(n-ary)演算子のセットすべてに適用される一般的なBoostPhoenix式ツリー変換を定義するにはどうすればよいですか。他を変更せずに残しますか?
いくつかのコードが役立つ場合があります。次のC++11コード(自動)を出力したいのですが、;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;
}