我正在尝试实现一个状态存储,它基本上是键值对的映射,其中值是std::variant
。
以下类工作正常。请注意,它不仅打印访问的值,还打印键:
class StateStoreTest
{
typedef std::variant<float, std::string> StateValue;
private:
std::map<int, StateValue> States;
public:
void SetValue(int key, const StateValue& value) { States[key] = value; }
void VisitAllValues()
{
for (auto& it : States)
{
auto key = it.first;
std::visit([key](auto&& value) { std::cout << key << "=" << value << "n"; }, it.second);
}
}
};
我可以这样使用它,输出符合预期:
StateStoreTest sst;
sst.SetValue(1, 123.4f);
sst.SetValue(2, "Test");
sst.VisitAllValues();
现在我不想在VisitAllValues()
中实现特定的访问者实现,但我希望能够提供可调用VisitAllValues()
方法的访问者。在该访客可调用对象中,我希望能够访问密钥和访问值。
所以基本上从呼叫者的角度来看,我希望能够做这样的事情:
sst.VisitAllValues([](auto key, auto&& value){ std::cout << key << "=" << value << "n"; });
我想我的VisitAllValues()
方法必须看起来像这样:
void VisitAllValues(??? visitor)
{
for (auto& it : States)
{
auto key = it.first;
std::visit([key](auto&& value) { visitor(key, value); }, it.second);
}
}
但是我无法弄清楚细节(我尝试将VisitAllValues()
编写为模板,但后来我在调用方遇到了问题(。
如何实现我的VisitAllValues()
方法,以便它接受同时有权访问 id 和值的访问者?
这是完整的工作玩具示例。
在我看来,如下应该有效
template <typename L>
void VisitAllValues (L const & visitor)
{
for (auto& it : States)
{
auto key = it.first;
std::visit([&](auto&& value) { visitor(key, value); }, it.second);
}
}
题外话:也许使用std::forward
是使用转发引用
visitor(key, std::forward<decltype(value)>(value));