定义一个符号,该符号可能是boost精神中文字功能的一部分
Defining a symbol which may be part of a literal function in boost spirit
我正在尝试读取一个数学函数,该函数依赖于带有boost::spirit的符号t
。
在下面的示例中,我尝试评估t=1.2
中的函数"tan(t)"
。而不是
Exit: 1, value = 2.5721
我得到
Exit: 1, value = 1.2
我知道,当我尝试读取函数"tan(t)"
时,不是计算t
的切线,而是将t
的值分配给单词tan
中的第一个字母。是否可以在不更改符号t
的情况下规避这种行为?此外,解析不应该失败吗?
#include <string>
#include <boost/spirit/include/phoenix_core.hpp>
#include <boost/config/warning_disable.hpp>
#include <boost/spirit/include/qi.hpp>
#include <boost/spirit/include/phoenix.hpp>
#include <boost/spirit/include/qi_symbols.hpp>
#include <boost/phoenix/stl/cmath.hpp>
namespace qi = boost::spirit::qi;
namespace ascii=boost::spirit::ascii;
using boost::spirit::ascii::space;
using boost::spirit::qi::symbols;
template< typename Iterator >
struct Grammar : public qi::grammar< Iterator, double(), ascii::space_type >
{
Grammar() : Grammar::base_type(expression)
{
using qi::double_;
using qi::_val;
using qi::_1;
expression = double_ [_val = _1]
| symbol [_val = _1]
| function [_val = _1]
| group [_val = _1];
function = qi::lit("tan") >> group [_val = boost::phoenix::tan(_1)];
group = '(' >> expression [_val = _1] >> ')' ;
}
qi::rule<Iterator, double(), ascii::space_type> expression, function, group;
qi::symbols<char, double > symbol;
};
int main()
{
typedef std::string::iterator iterator;
Grammar<iterator> grammar;
std::string function = "tan(t)"; //it would work if function = "tan(x)"
grammar.symbol.add("t",1.2); // and add("x",1.2)
double value;
bool r = qi::phrase_parse(function.begin(), function.end(), grammar, space, value);
std::cout << "Exit: " << r << ", value = " << value << std::endl;
return 0;
}
您必须重新排序规则。你的符号(t
)正在吞噬tan
的第一个字母。所以,您实际上根本没有解析所有输入!
如果启用调试,您会看到以下输出:
<expression>
<try>tan(t)</try>
<success>an(t)</success>
<attributes>[1.2]</attributes>
</expression>
Exit: 1, value = 1.2
解决这个问题的"皇家方法"是使用Spirit Repository中的Qi Distinct Keyword指令:boost::Spirit::Qi关键字和标识符
在Coliru上直播
#define BOOST_SPIRIT_DEBUG
#define BOOST_SPIRIT_USE_PHOENIX_V3
#include <boost/spirit/include/phoenix_core.hpp>
#include <boost/config/warning_disable.hpp>
#include <boost/spirit/include/qi.hpp>
#include <boost/spirit/include/phoenix.hpp>
#include <boost/spirit/include/qi_symbols.hpp>
#include <boost/spirit/repository/include/qi_distinct.hpp>
#include <boost/phoenix/stl/cmath.hpp>
namespace qi = boost::spirit::qi;
namespace ascii=boost::spirit::ascii;
using boost::spirit::ascii::space;
using boost::spirit::qi::symbols;
template< typename Iterator >
struct Grammar : public qi::grammar< Iterator, double(), ascii::space_type >
{
Grammar() : Grammar::base_type(expression)
{
using qi::double_;
using qi::_val;
using qi::_1;
using boost::spirit::repository::qi::distinct;
expression = double_
| distinct(qi::char_("a-zAZ09_")) [ symbol ]
| function
| group;
function = "tan" >> group [_val = boost::phoenix::tan(_1)];
group = '(' >> expression >> ')' ;
BOOST_SPIRIT_DEBUG_NODES((expression)(function)(group));
}
qi::rule<Iterator, double(), ascii::space_type> expression, function, group;
qi::symbols<char, double > symbol;
};
int main()
{
typedef std::string::iterator iterator;
Grammar<iterator> grammar;
std::string function = "tan(t)";
grammar.symbol.add("t",1.2);
double value;
bool r = qi::phrase_parse(function.begin(), function.end(), grammar, space, value);
std::cout << "Exit: " << r << ", value = " << value << std::endl;
return 0;
}
带有调试信息的输出为:
<expression>
<try>tan(t)</try>
<function>
<try>tan(t)</try>
<group>
<try>(t)</try>
<expression>
<try>t)</try>
<success>)</success>
<attributes>[1.2]</attributes>
</expression>
<success></success>
<attributes>[1.2]</attributes>
</group>
<success></success>
<attributes>[2.57215]</attributes>
</function>
<success></success>
<attributes>[2.57215]</attributes>
</expression>
Exit: 1, value = 2.57215
相关文章:
- 如何打印boost多精度128位无符号整数
- 如何修复架构x86_64的未定义符号,Boost Asio 1.58
- boost::任何带有结构体和无符号整数
- 使用 Boost/Python 的未定义符号 - 复杂
- boost::存在符号链接不取消引用
- 如何使用 boost asio 通过 UDP 套接字发送无符号字符的 std::vector?
- 无符号的 int 到 IP 地址字符串,不带 itoa/to_string/boost
- 使用 Boost.Python 修复未解析的外部符号
- boost.mpi给出了架构x86_64的未定义符号
- 使用Boost/Python的未定义符号
- OS X, CMake, Boost: 未定义的架构符号 x86_64: "boost::system::detail::generic_category_instance"
- 使用 boost 解析符号链接时,结果不等于原始路径名
- 打印带有 boost::format 的布尔值作为符号值
- 访问Boost Bind时,共享库中的C 符号查找错误
- 使用 Boost.Python 使用无符号字符和参数公开方法
- Boost Boost_program_options-gcc41-mt_1-39.在进行静态链接时出现未定义的符号错误
- 我的Py_NoneStruct符号(python,boost.python)在哪里
- Boost::python带有ACE,C++符号不明确
- 在C++中使用 boost:regex_error 时未定义的符号
- boost::bind 似乎使用不同的编译器生成不同的符号