/*============================================================================= Copyright (c) 2001-2003 Andy Elvey Copyright (c) 2001-2003 Dan Nuffer http://spirit.sourceforge.net/ Permission to copy, use, modify, sell and distribute this software is granted provided this copyright notice appears in all copies. This software is provided "as is" without express or implied warranty, and with no claim as to its suitability for any purpose. =============================================================================*/ // // A very simple parser grammar . // This parser parses a simple polynomial expression ( of the form // aX^2 + bX^3 + ... ) . // // Uses: The Spirit parser framework, which was written by // Joel de Guzman // //#define BOOST_SPIRIT_DEBUG ///$$$ DEFINE THIS WHEN DEBUGGING $$$/// #include #include #include #include #include /////////////////////////////////////////////////////////////////////////////// using namespace boost::spirit; using namespace std; /////////////////////////////////////////////////////////////////////////////// //---------------------------------------------------------------------------- // Start grammar definition //---------------------------------------------------------------------------- struct polynomial_grammar : public grammar { template struct definition { typedef rule rule_t; rule_t program, stmts, term, constant, sign, var, caret; rule_t IDENT, STRING_LITERAL, VAR; definition(polynomial_grammar const&) { //----------------------------------------------------------------- // OPERATORS //----------------------------------------------------------------- chlit<> PLUS('+'); chlit<> MINUS('-'); chlit<> CARET('^'); //----------------------------------------------------------------- // TOKENS //----------------------------------------------------------------- IDENT = as_lower_d [ lexeme_d [ (alpha_p >> *(alnum_p | '_')) ] ] ; STRING_LITERAL = lexeme_d [ ch_p('\'') >> +( str_p("\'\'") | anychar_p - ch_p('\'') ) >> ch_p('\'') ] ; VAR = as_lower_d[ lexeme_d[ alpha_p ] ] ; //----------------------------------------------------------------- // RULES //----------------------------------------------------------------- // Now - the actual BNF grammar for the parser program = +(stmts) ; stmts = ( term | constant ) ; term = +( !(sign) >> !(uint_p) >> VAR >> !(CARET) >> !(uint_p) ) ; // e.g. 3x^2 constant = !(sign) >> +(uint_p) ; sign = ( PLUS | MINUS ) ; var = VAR ; caret = CARET ; } rule_t const& start() const { return program; } }; }; int main(int /*argc*/, char* /*argv[]*/) { cout << "/////////////////////////////////////////////////////////\n\n"; cout << "\t\t A polynomial parser...\n\n"; cout << "/////////////////////////////////////////////////////////\n\n"; cout << "Type a polynomial (e.g. 3a^4 + 7a^5 - 8a^3 - 4a - 5) or\n" << "[q or Q] to quit\n\n"; polynomial_grammar g; string str; while (getline(cin, str)) { if (str[0] == 'q' || str[0] == 'Q') break; if (parse(str.c_str(), g, space_p).full) { cout << "parsing succeeded\n"; } else { cout << "parsing failed\n"; } } cout << "Bye... :-) \n\n"; return 0; }