- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我需要使用 Boost::Spirit 将如下所示的中缀表示法转换为 n 元前缀表示法,但我无法基于 https://stackoverflow.com/a/8707598/1816477 的答案进行构建等
这就是我要解析的内容:
not (xyz='a' or xyz='b' or xyz='c') and abc='s' xor (pqr ='v' and xyz='d')
这个 LISP 风格的格式是我试图提供的输出(不要介意缩进):
(xor (and (= pqr 'v') (= xyz 'd'))
(and (= abc 's')
(not (or (= xyz 'a')
(= xyz 'b')
(= xyz 'c')))))
因此,我尝试解析的术语由前缀表达式(not <expression>
)和中缀表达式(<expression> and <expression> and ...
等)组成,即:赋值、否定和 n 元 ands、ors、xors 等,暗示运算符优先级(或 < xor < 和 < assignment < 否定)。
我不擅长的是语法正确。输出到合适的 boost::variant
代表我认为我能够完成的已解析 bool 表达式。我正在考虑这样的输出结构:
struct prefixExpr;
struct infixExpr;
typedef boost::variant<
std::string, // identifiers, values etc.
boost::recursive_wrapper<prefixExpr>, // e.g. negation
boost::recursive_wrapper<infixExpr> // assignment, and, or, xor etc.
> expression;
struct prefixExpr {
std::string op; // currently only "not"
expression expr;
};
BOOST_FUSION_ADAPT_STRUCT(prefixExpr, op, expr)
struct infixExpr {
std::string op; // "and", "or", "xor", "="
std::vector<expression> exprs;
};
BOOST_FUSION_ADAPT_STRUCT(infixExpr, op, exprs)
我需要做什么才能像上面提到的那样解析表达式并将它们转换为前缀表示法?
我使用的是 boost 1.67.0(撰写本文时最新版本)和 Visual Studio 15.7.3(也是撰写本文时最新版本)。
最佳答案
代码并不完美,但应该很容易理解:
#include <boost/variant.hpp>
#include <boost/spirit/home/x3.hpp>
#include <vector>
#include <string>
#include <iostream>
struct id : std::string {};
struct value : std::string {};
struct nary_expr;
using expr = boost::variant<
id, value,
boost::recursive_wrapper<nary_expr>
>;
struct nary_expr
{
std::string op;
std::vector<expr> exprs;
};
namespace x3 = boost::spirit::x3;
auto compose_nary_expr = [](auto& ctx)
{
//auto&& [left, tail] = x3::_attr(ctx);
auto&& left = boost::fusion::at_c<0>(x3::_attr(ctx));
auto&& tail = boost::fusion::at_c<1>(x3::_attr(ctx));
if (tail.size() == 0) {
x3::_val(ctx) = left;
return;
}
// left associativity
auto op = boost::fusion::at_c<0>(tail[0]);
std::vector<expr> exprs = { left, boost::fusion::at_c<1>(tail[0]) };
for (std::size_t i = 1; i < tail.size(); ++i) {
// same priority but different operator
auto&& next_op = boost::fusion::at_c<0>(tail[i]);
if (op != next_op) {
exprs = std::vector<expr>{ nary_expr{ op, std::move(exprs) } };
op = next_op;
}
exprs.push_back(boost::fusion::at_c<1>(tail[i]));
}
x3::_val(ctx) = nary_expr{ op, std::move(exprs) };
};
x3::rule<class prec4_expr_rule, expr> const prec4_expr("prec4_expr");
x3::rule<class prec3_expr_rule, expr> const prec3_expr("prec3_expr");
x3::rule<class prec2_expr_rule, expr> const prec2_expr("prec2_expr");
x3::rule<class prec1_expr_rule, expr> const prec1_expr("prec1_expr");
x3::rule<class prec0_expr_rule, expr> const prec0_expr("prec0_expr");
auto const prec4_expr_def = prec4_expr = (
prec3_expr
>> *( (x3::string("or") > prec3_expr)
)
)[compose_nary_expr];
auto const prec3_expr_def = prec3_expr = (
prec2_expr
>> *( (x3::string("xor") > prec2_expr)
)
)[compose_nary_expr];
auto const prec2_expr_def = prec2_expr = (
prec1_expr
>> *( (x3::string("and") > prec1_expr)
)
)[compose_nary_expr];
auto compose_binary_expr = [](auto& ctx)
{
auto&& rhs = boost::fusion::at_c<0>(x3::_attr(ctx));
auto&& tail = boost::fusion::at_c<1>(x3::_attr(ctx));
if (tail.size() > 0) {
auto&& op = boost::fusion::at_c<0>(tail[0]);
auto&& lhs = boost::fusion::at_c<1>(tail[0]);
x3::_val(ctx) = nary_expr{ op, { rhs, lhs } };
}
else {
x3::_val(ctx) = rhs;
}
};
// should use optional, but something wrong with spirit
auto const prec1_expr_def = prec1_expr = (
prec0_expr >> *(x3::string("=") > prec0_expr)
)[compose_binary_expr];
x3::rule<class not_expr_rule, expr> const not_expr("not_expr");
auto compose_unary_expr = [](auto& ctx)
{
//auto&& [op, expr] = x3::_attr(ctx);
auto&& op = boost::fusion::at_c<0>(x3::_attr(ctx));
auto&& expr = boost::fusion::at_c<1>(x3::_attr(ctx));
x3::_val(ctx) = nary_expr{ op, { expr } };
};
auto const not_expr_def = not_expr = (x3::string("not") > prec0_expr)[compose_unary_expr];
auto const id_term = x3::rule<class id_r, id>{} = x3::lexeme[x3::alpha >> *x3::alnum];
auto const value_term = x3::rule<class value_r, value>{} = x3::lexeme["'" > +~x3::char_('\'') >> "'"];
auto const prec0_expr_def =
value_term
| ( '(' > prec4_expr >> ')' )
| not_expr
| id_term
;
BOOST_SPIRIT_DEFINE(
prec0_expr
, prec1_expr
, prec2_expr
, prec3_expr
, prec4_expr
, not_expr
);
struct indent
{
std::size_t cur;
};
indent operator+(indent lhs, std::size_t rhs)
{
return { lhs.cur + rhs };
}
std::ostream& operator<<(std::ostream& os, indent const& v)
{
for (unsigned i = 0; i < v.cur; ++i) os << ' ';
return os;
}
struct is_simple
{
template <typename T>
bool operator()(T const&) const
{
return std::is_same<T, id>::value || std::is_same<T, value>::value;
}
};
struct printer
{
indent indent_;
void operator()(id const& v)
{
std::cout << v;
}
void operator()(value const& v)
{
std::cout << '\'' << v << '\'';
}
void operator()(nary_expr const& v)
{
std::cout << '(' << v.op << ' ';
printer p{ indent_ + 2 + v.op.size() };
boost::apply_visitor(p, v.exprs[0]);
for (std::size_t i = 1; i < v.exprs.size(); ++i) {
if (boost::apply_visitor(is_simple{}, v.exprs[i])) {
std::cout << ' ';
}
else {
std::cout << '\n' << p.indent_;
}
boost::apply_visitor(p, v.exprs[i]);
}
std::cout << ')';
}
};
int main()
{
std::string s = "not (xyz='a' or xyz='b' or xyz='c') and abc='s' xor (pqr ='v' and xyz='d')";
expr expr;
auto iter = s.cbegin();
if (phrase_parse(iter, s.cend(), prec4_expr_def, x3::space, expr) && iter == s.cend()) {
boost::apply_visitor(printer{}, expr);
}
return 0;
}
它打印:
(xor (and (not (or (= xyz 'a')
(= xyz 'b')
(= xyz 'c')))
(= abc 's'))
(and (= pqr 'v')
(= xyz 'd')))
关于c++ - 使用 Boost::Spirit 从中缀到前缀的 n 元 bool 语法转换?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/50843249/
我有一个带有列的表提供者 implied(tiny int)(something like nullable bool) provi
我正在阅读 VideoFileWriter来自 AForge.Video.FFMPEG 的类(class)通过 ILSPY 组装(我很想看看特定方法是如何工作的)并发现了这个: public bool
这是我的完整代码... import 'package:flutter/cupertino.dart'; import 'package:flutter/material.dart'; import
我有一个输入 list类型 [Maybe SomeType]和一个谓词 p类型 SomeType -> Bool ,我想回答这个问题“谓词 p 是否适用于所有碰巧在输入中的 SomeType ?”。
使用 !!x 有什么区别吗?对比(bool)x ? 假设__STDC_VERSION__ >= 199901L和 #include 他们都保证结果是0吗?或 1 ,并且无论 x 的大小和值如何,都不
我正在编写一些 C++ 代码,我想调用两个函数(checkXDirty 和 checkYDirty),并返回 true如果任一返回 true。即使一个返回 true 我也需要评估两者,所以我的第一个想
我注意到 bool在 QtCreator 中以不同于其他类型的颜色突出显示: 只有在包含某些 header 时才会发生这种情况,最终我将其追踪到 . QtCreator 的代码检查器似乎无法手动跟踪
有一个函数: func (first: Int) -> Int -> Bool -> String { return ? } 返回值怎么写?我对上面 func 的返回类型感到很困惑。 最
训练神经网络学习“异或” 我正在尝试使用“批量归一化”,我创建了一个批量归一化层函数“batch_norm1”。 import tensorflow as tf import nump
我已经创建了任务函数来验证我的 json 文件。一切正常,直到我没有使用结果。当我试图从 async task function 获得结果时它显示错误为 Cannot implicitly conve
我有一个函数 func login (parameters: [(String, Any)], completion: @escaping (Bool) -> Vo
我正在处理最近从 X/Motif 转移到 Qt 的 C++ 代码库。我正在尝试编写一个 Perl 脚本,它将用 bool 替换所有出现的 Boolean(来自 X)。该脚本只是做了一个简单的替换。 s
嗨,我正尝试创建一个Visiblity小部件,如果用户在Firebase数据库阵列上,该小部件将显示。看起来像这样(成员数组): 如您所见,我创建了一个StreamBuilder,如果当前用户的用户名
我创建了如下的rest api方法, Future activateAccount(int id, int code) async{ final body = {"code": '$c
在我的Flutter应用中,我有一个返回Future的函数,但我想将结果作为Stream。这是函数: Future isGpsOn() async { if (await Geolocat
我可以看到 BOOLEAN 覆盖了 __visit_name__ class BOOLEAN(Boolean): __visit_name__ = 'BOOLEAN' 控制调度员选择的访问者方
考虑以下代码: bool x; bool? y = null; x = y?? true; 将 bool? 分配给 bool 是一个编译时错误,但上面的代码在编译和运行时都成功了。为什么?尽管第三条语
我正在重写一些 Javascript 代码以在 Excel VBA 中工作。由于在这个网站上搜索,我已经设法翻译了几乎所有的 Javascript 代码!但是,有些代码我无法准确理解它在做什么。这是一
我想拍一张bool来自Vec并在 if 语句中进行比较。如何解决以下错误? | 7 | if cell { | ^^^^ expected
我在我的应用程序崩溃跟踪工具中发现了一些崩溃。基本上我有一个 tabBarController,其中一个选项卡有一个嵌入式 UIWebView,另一个选项卡有一个带有 UITableView 的 Co
我是一名优秀的程序员,十分优秀!