Как использовать класс только с одним атрибутом в AST с Boost Spirit?
Я хочу проанализировать файл в AST, используя Boost Spirit.
Корнем моего AST является класс с одним атрибутом:
typedef boost::variant<FunctionDeclaration, GlobalVariableDeclaration> FirstLevelBlock;
struct Program {
std::vector<FirstLevelBlock> blocks;
};
BOOST_FUSION_ADAPT_STRUCT(
::Program,
(std::vector<eddic::FirstLevelBlock>, blocks)
)
Если я анализирую, используя одно правило:
program %= *(function | globalDeclaration);
он не компилируется, но если я добавлю одно строковое имя в программу, он будет работать хорошо. Я мог бы использовать вектор как корень, но я хочу использовать класс, потому что я хочу добавить некоторые методы в класс Program.
РЕДАКТИРОВАТЬ:
Если я окружаю свою программу фигурными скобками, она работает хорошо:
program %= lexer.left_brace >> *(function | globalDeclaration) >> lexer.right_brace;
компилируется и работает нормально, но:
program %= *(function | globalDeclaration);
не компилируется...
Есть ли в Boost Spirit что-то, что мешает использовать такие простые правила?
1 ответ
Отредактированный вопрос версия 2
Если я окружаю свою программу фигурными скобками, она хорошо работает [...], но
program %= *(function | globalDeclaration);
не компилируется...Есть ли в Boost Spirit что-то, что мешает использовать такие простые правила?
Во-первых, мы не можем сказать, без определения function
а также globalDeclaration
,
Во-вторых, я попытался изменить мои строки PoC на
static const qi::rule<It, Program(), space_type> program = *(function | global);
Program d = test("void test(); int abc; int xyz; void last();" , program);
И вот, я получаю ошибку вашего компилятора! Теперь я, безусловно, согласен, что это очень похоже на ошибку преобразования атрибутов. Также вот предварительный обходной путь:
program %= eps >> *(function | global);
Как вы видете, qi::eps
в помощь
Ответ на оригинальный вопрос версии 1
Ммм. Я думаю, вам нужно опубликовать минимальный рабочий образец. Вот подтверждение концепции, начиная с вашего вопроса, и все это работает довольно хорошо.
Обратите внимание, что я скомпилирован с g++ -std=c++0x
для того, чтобы получить по умолчанию Attr
аргумент параметра на test
функция.
#include <boost/spirit/include/qi.hpp>
#include <boost/spirit/include/karma.hpp>
#include <boost/fusion/adapted.hpp>
#include <boost/strong_typedef.hpp>
// added missing bits
namespace eddic
{
typedef std::string FunctionDeclaration;
typedef std::string GlobalVariableDeclaration;
typedef boost::variant<FunctionDeclaration, GlobalVariableDeclaration> FirstLevelBlock;
}
using namespace eddic;
// end missing bits
struct Program {
std::vector<FirstLevelBlock> blocks;
};
BOOST_FUSION_ADAPT_STRUCT(
::Program,
(std::vector<eddic::FirstLevelBlock>, blocks)
)
namespace /*anon*/
{
using namespace boost::spirit::karma;
struct dumpvariant : boost::static_visitor<std::ostream&>
{
dumpvariant(std::ostream& os) : _os(os) {}
template <typename T> std::ostream& operator ()(const T& t) const
{ return _os << format(stream, t); }
private: std::ostream& _os;
};
std::ostream& operator<<(std::ostream& os, const FirstLevelBlock& block)
{
os << "variant[" << block.which() << ", ";
boost::apply_visitor(dumpvariant(os), block);
return os << "]";
}
std::ostream& operator<<(std::ostream& os, const std::vector<FirstLevelBlock>& blocks)
{ return os << format(-(stream % eol), blocks); }
std::ostream& operator<<(std::ostream& os, const Program& program)
{ return os << "BEGIN\n" << program.blocks << "\nEND"; }
}
namespace qi = boost::spirit::qi;
template <typename Rule, typename Attr = typename Rule::attr_type>
Attr test(const std::string& input, const Rule& rule)
{
typedef std::string::const_iterator It;
It f(input.begin()), l(input.end());
Attr result;
try
{
bool ok = qi::phrase_parse(f, l, rule, qi::space, result);
if (!ok)
std::cerr << " -- ERR: parse failed" << std::endl;
} catch(qi::expectation_failure<It>& e)
{
std::cerr << " -- ERR: expectation failure at '" << std::string(e.first, e.last) << "'" << std::endl;
}
if (f!=l)
std::cerr << " -- WARN: remaing input '" << std::string(f,l) << "'" << std::endl;
return result;
}
int main()
{
typedef std::string::const_iterator It;
static const qi::rule<It, FunctionDeclaration(), space_type> function = "void " > +~qi::char_("()") > "();";
static const qi::rule<It, GlobalVariableDeclaration(), space_type> global = "int " > +~qi::char_(";") > ";";
static const qi::rule<It, FirstLevelBlock(), space_type> block = function | global;
static const qi::rule<It, Program(), space_type> program = '{' >> *(function | global) >> '}';
FunctionDeclaration a = test("void test();", function);
std::cout << "FunctionDeclaration a : " << a << std::endl;
GlobalVariableDeclaration b = test("int abc;", global);
std::cout << "GlobalVariableDeclaration b : " << b << std::endl;
FirstLevelBlock c = test("void more();", block);
std::cout << "FirstLevelBlock c : " << c << std::endl;
/*FirstLevelBlock*/ c = test("int bcd;", block);
std::cout << "FirstLevelBlock c : " << c << std::endl;
Program d = test("{"
"void test();"
"int abc"
";"
"int xyz; void last();"
"}", program);
std::cout << "Program d : " << d << std::endl;
}
Выход:
FunctionDeclaration a : test
GlobalVariableDeclaration b : abc
FirstLevelBlock c : variant[1, more]
FirstLevelBlock c : variant[1, bcd]
Program d : BEGIN
test
abc
xyz
last
END