动态表达
如果你想从用户那里收到一个字符串并从中构建一个表达式,那么C++数学表达式库可能适合您的账单吗?
templatevoid trig_function() { std::string expression_string = "clamp(-1.0,sin(2 * pi * x) + cos(x / 2 * pi),+1.0)"; T x; exprtk::symbol_table symbol_table; symbol_table.add_variable("x",x); symbol_table.add_constants(); exprtk::expression expression; expression.register_symbol_table(symbol_table); exprtk::parser parser; parser.compile(expression_string,expression); for (x = T(-5.0); x <= T(+5.0); x += 0.001) { T y = expression.value(); printf("%19.15f\t%19.15f\n",x,y); } }
还有可能嵌入脚本语言,例如Lua或Python,它将为您提供(甚至)更多功能.如果您正在编写游戏,则需要考虑这一点,因为您可能希望编写大部分内容.
如果您正在使用Qt,则可以使用QtScript(Javascript-ish)运行从QObject派生对象读取(静态或动态)属性的表达式.
使用上面的一个使你不必编写自己的解析器,AST和评估器,但是对于一小组操作符,如果你使用Boost.Spirit或其他一些像样的解析库,那么攻击它们应该不会太难.
静态表达式
要在一组预定义表达式之间进行选择(即在编译时已知),您应该将表达式存储在多态函数对象中.
对于C++ 11,如果您可以使用它,请使用std::function
和lambda表达式.
std::functionexpr = [](int a, int b) { a*2 < b };
对于早期的编译器,我推荐使用函数和绑定,在Boost(boost::)或C++ 0x TR1(std::)中,具体取决于您的编译器.此外,Boost.Lambda将在这里提供帮助,因为它允许您构造和存储表达式以供以后评估.但是,如果您不熟悉C++和模板(或函数式编程),它可能会让您感到非常恐慌.
有了这个,你可以写
using namespace boost::lambda; boost::functionmyexpr1 = (_1 + _2) > 20; boost::function myexpr2 = (_1 * _2) > 42; std::cout << myexpr1(4,7) << " " << myexpr2(2,5);
与绑定,它看起来像:
boost::functioncheck = bind(&Player::getHealth, _1) > 20; Player p1; if (check(p1)) { dostuff(); } check = bind(&Player::getGold, _1) < 42; if (check(p1)) { doOtherStuff(); }