0

我有一个这样的数据结构:map<string, map<string, map<string, MyObj>>>

现在,我有多个函数,它们都使用相同的 for 循环方法:

for (auto p1 : myMap)
    for (auto p2 : p1.second)
        for (auto p3 : p2.second)
            doThingsWith(p1, p2, 3);

函数之间的doThingsWith(p1, p2, p3)差异以及 for 循环之前和之后的代码。此外,一些函数例如只需要访问MyObj对象,而另一些函数需要访问所有字符串键以及MyObj对象。

那么,问题是,是否有任何方法可以在不损失性能的情况下对其进行概括?我想出了一个返回元组向量的函数:

vector<tuple<string, string, string, MyObj>> getData(... &myMap)
{
    vector<tuple<string, string, string, MyObj>> data;
    for (auto p1 : myMap)
        for (auto p2 : p1.second)
            for (auto p3 : p2.second)
                data.push_back(tuple<string, string, string, MyObj>(
                    p1.first, p2.first, p3.first, p3.second
                ));
    return data;
}

现在我的函数可以使用这个:

for (auto t : getData(myMap))
    doThingsWith(get<0>(t), get<1>(t), get<2>(t), get<3>(t));

但这不必要地构造了很多元组和向量,因为myMap它很大。

有没有更好的办法?在 Python 中,我可以使用生成器,但我不知道 C++ 等价物:

def iterTuples(myMap):
    for k1, v1 in myMap.items():
        for k2, v2 in v1.items():
            for k3, v3 in v2.items():
                yield k1, k2, k3, v3


for k1, k2, k3, val in iterTuples(myMap):
    doThingsWith(k1, k2, k3, val)
4

2 回答 2

3

您的 map-of-maps-of-maps 从一开始就没有效率。最好是整合关键部分:

typedef tuple<string, string, string> Key;
std::map<Key, MyObj> myMap;

现在您有了一个简单有效的解决方案:

for (const auto& pr : myMap)
    doThingsWith(get<0>(pr.first), get<1>(pr.first), get<2>(pr.first), pr.second);

而且你避免了很多间接。如果您可以使用固定长度的字符串和/或将字符串组合到单个分配中,则更是如此。或者,如果共享第一个和第二个键字符串很重要,您可以使用引用计数字符串类型。

于 2018-02-21T12:29:12.383 回答
0

只需创建模板函数:

template <typename Map3, typename F>
void ForEachAll(Map3&& m, F&& f)
{
    for (auto&& p1 : m)
        for (auto&& p2 : p1.second)
            for (auto&& p3 : p2.second)
                f(p1.first, p2.first, p3.first, p3.second);
}

template <typename Map3, typename F>
void ForEach(Map3&& m, F&& f)
{
    for (auto&& p1 : m)
        for (auto&& p2 : p1.second)
            for (auto&& p3 : p2.second)
                f(p3.second);
}

笔记:

  • 你可以用一些 SFINAE 来命名它们
  • MAP3是允许 const 和没有 const 映射的模板。
于 2018-02-21T12:28:20.473 回答