5

如果我有一个符号表:

struct MySymbols : symbols<char, MyEnum::Fruits>
{
    MySymbols ()
        : symbols<char, MyEnum::Fruits>(std::string("MySymbols"))
    {
        add("apple", MyEnum::Apple)
           ("orange", MyEnum::Orange);
    }
};

我想遍历表以便按数据值搜索符号。我不能使用 lambda 表达式,所以我实现了一个简单的类:

template<typename T>
struct SymbolSearcher
{
    SymbolSearcher::SymbolSearcher(T searchFor)
        : _sought(searchFor)
    {
        // do nothing
    }

    void operator() (std::basic_string<char> s, T ct)
    {
        if (_sought == ct)
        {
            _found = s;
        }
    }

    std::string found() const { return _found; }

private:
    T _sought;
    std::string _found;
};

我使用它如下:

SymbolSearcher<MyEnum::Fruits> search(ct);
MySymbols symbols;

symbols.for_each(search);
std::string symbolStr = search.found();

如果我设置断点,_found = s我可以确认 _found 正在设置,但是 search.found() 总是返回一个空字符串。我猜这与在 for_each 中调用仿函数的方式有关,但我不知道。

我究竟做错了什么?

4

1 回答 1

5

可能是这样

  • 字符串的实际值是空字符串(不太可能)

  • 函子按值传递,使有状态函子无用(因为原始状态实际上不会被传递)。

您可以将该_found字段作为参考(要求您确保遵守三规则以使其发挥作用)。

这是一个演示,通过断言往返结果来显示原理:http SymbolSearcher: //liveworkspace.org/code/4qupWC$1

#include <boost/spirit/include/qi.hpp>

namespace qi     = boost::spirit::qi;

template<typename T>
struct SymbolSearcher
{
    SymbolSearcher(T searchFor, std::string& result) : _sought(searchFor), _found(result)
    {
    }

    void operator() (std::basic_string<char> s, T ct)
    {
        if (_sought == ct)
        {
            _found = s;
        }
    }

    std::string found() const { return _found; }

private:
    T _sought;
    std::string& _found;
};

int main()
{
    const std::string str("mies");

    typedef std::string::const_iterator It;
    It begin = str.cbegin();
    It end   = str.cend();

    qi::symbols<char, int> symbols;
    symbols.add("aap", 1)("noot", 2)("mies", 3);

    int out;
    bool ok = qi::parse(begin, end, symbols, out);
    assert(ok);

    std::string found;
    SymbolSearcher<int> sf(out, found);
    symbols.for_each(sf);

    assert(str == sf.found());
}
于 2013-02-27T01:26:24.987 回答