【问题标题】:Is it possible to use mpl::map to initialize symbol parser?是否可以使用 mpl::map 来初始化符号解析器?
【发布时间】:2015-01-26 06:33:02
【问题描述】:

如果我有 mpl::map,如何生成对应的 boost::spirit::symbol 解析器?

例子:

using blocks = mpl::map<
                  mpl::pair<mpl::string<'p'>, do_para>,
                  mpl::pair<mpl::string<'ul'>, do_ul>,
                  mpl::pair<mpl::string<'ol'>, do_ol>
               >;

 qi::symbols<const char *, T> block_parser(?????);

感谢有关最小代码问题的帮助!

【问题讨论】:

  • 您基本上是在问“可以使用 mpl 映射来产生运行时效果”吗?嗯,很明显。我的问题是:为什么(因为它似乎没有更多表现力,更可维护......?)
  • 经过几次其他 mpl 转换后,我得到了 mpl 映射。可能仍然不是正确的方法,但不认为它应该是死胡同。

标签: c++ boost-spirit boost-mpl


【解决方案1】:

在不知道你为什么会做这样的事情的情况下,这是一个使用boost::fusion::for_each的想象应用:

Live On Coliru

#include <boost/mpl/map.hpp>
#include <boost/mpl/for_each.hpp>
#include <boost/mpl/string.hpp>
#include <boost/spirit/include/qi.hpp>
#include <boost/fusion/algorithm.hpp>
#include <boost/fusion/mpl.hpp>

namespace mpl = boost::mpl;
namespace qi = boost::spirit::qi;

struct do_ol   : qi::grammar<const char*> {
    do_ol() : do_ol::base_type(start) {}
    qi::rule<const char*> start;
};
struct do_ul   : qi::grammar<const char*> {
    do_ul() : do_ul::base_type(start) {}
    qi::rule<const char*> start;
};
struct do_para : qi::grammar<const char*> {
    do_para() : do_para::base_type(start) {}
    qi::rule<const char*> start;
};

template <typename T>
struct block_parser_t : qi::symbols<char, T> {

    template <typename Map>
    void add_map() {
        boost::fusion::for_each(Map(), map_adder(*this));
    }

  private:
    struct map_adder {
        map_adder(block_parser_t& r) : _r(r) {}
        block_parser_t& _r;

        template <typename...> struct result { typedef void type; };

        template <typename Pair> void operator()(Pair const&) const { 
            std::cout << "Adding: " << mpl::c_str<typename Pair::first>::value << "\n";
            _r.add(
                mpl::c_str<typename Pair::first>::value,
                typename Pair::second()
            );
        }
    };
};

int main() {
    using blocks = mpl::map<
        mpl::pair<mpl::string<'p'>,  do_para>,
        mpl::pair<mpl::string<'ul'>, do_ul>,
        mpl::pair<mpl::string<'ol'>, do_ol>
    >;

    typedef qi::rule<char const*> R;

    block_parser_t<R> block_parser;
    block_parser.add_map<blocks>();
}

打印

Adding: p
Adding: ul
Adding: ol

它将默认构造do_uldo_oldo_para语法(假设它们是语法)

【讨论】:

  • 感谢工作示例!我被“map_adder”函子困住了——我试图从参数中获取数据,而不是从类型中获取数据。我想我越过运行时边界有点太早了...再次感谢您的帮助!
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 2020-11-13
  • 2016-03-03
  • 2011-01-26
  • 2021-12-26
  • 2016-04-26
  • 2016-12-20
相关资源
最近更新 更多