有没有STL函数可以得到两个C++向量的笛卡尔乘积
Is there some STL function to get cartesian product of two C++ vectors?
假设
b = ["good ", "bad "]
a = ["apple","mango"]
then output = ["good apple","good mango","bad apple","bad mango"]
我知道这可以用嵌套的for循环来完成,但有没有一些优雅的一行代码可以用C++STL来完成呢?
这里有一行(复制自Jonathan Mee在这里发布的答案):
for(size_t i = 0, s = a.size(); i < output.size(); ++i) output[i] = b[i/s] + ' ' + a[i%s];
这里是完整的示例。
给定vector<string> a
和vector<string> b
,您可以使用for_each
:
vector<string> output(size(a) * size(b));
for_each(begin(output), end(output), [&, it = 0U](auto& i) mutable {
i = a[it / size(b)] + ' ' + b[it % size(b)];
++it;
});
实时示例
编辑:
我们已经初始化了output
,有足够的空间容纳a
和b
的每一个组合。然后我们将遍历output
的每个元素并分配它
我们将使用a
的1st元素作为output
的第一个size(b)
元素,使用a
的2nd元件作为第二个size(b)
元素,依此类推。因此,我们将通过使用it / size(b)
进行索引来实现这一点。我们希望通过b
元素的迭代将其结合起来。
it
将移动到output
的每个元素的下一个索引,但索引需要换行,否则当it == size(b)
时它将越界,为此我们使用it % size(b)
。
第2版:
在这个问题中,通过基准测试,我发现了一个现象,即模和除法是迭代的昂贵操作。我在这里做了同样的测试。为了隔离算法,我只是在vector<int>
而不是vector<string>
上进行笛卡尔求和。
首先,我们可以看到这两种算法导致不同的组装。我上面写的算法需要585行汇编。根据我对MSalter代码的解释,需要588行
vector<string> output(size(testValues1) * size(testValues2));
auto i = begin(output);
std::for_each(cbegin(a), cend(a), [&](const auto& A) { std::for_each(cbegin(b), cend(b), [&](const auto& B) { *i++ = A + ' ' + B; }); });
我在这里放置了一个相当可靠的基准测试:http://ideone.com/1YpzIO在测试中,我只设置了100次测试,但MSchanges的算法总是获胜。在本地使用Visual Studio 2015发布的10000000次测试中,MSchanges算法的完成时间大约是我的三分之二。
很明显,模不是一个很好的索引方法:(
没有直接的解决方案;我检查了整个<algorithm>
。没有一个函数产生长度为M*N的输出。
可以调用第一个范围上的std::for_each
,使用在第二个范围上调用std::for_each
的lambda(!)
std::vector<std::string> a, b;
std::for_each(a.begin(), a.end(),
[&](std::string A) { std::for_each(b.begin(), b.end(),
[A](std::string B) { std::cout << A << '/' << B << 'n'; }
);});
但这只是STL中的一个嵌套循环。
- 如何在C++中从两个单独的for循环中添加两个数组
- 为什么两个不同的未命名名称空间可以共存于一个cpp文件中
- 当在同一名称空间中有两个具有相同签名的函数时,会发生什么
- 如何返回一个类的两个对象相加的结果
- 如何在C++中将一个无符号的 int 转换为两个无符号的短裤?
- 如何将两个不同矢量的同一位置的两个元素组合在一起
- 两个字符串在 c++ 中不相等
- 在两个类中共享相同的函数调用,并在不需要时避免空实例化
- 两个文件使用彼此的功能-如何解决
- 为什么Mat类的两个对象可以在不重载运算符+的情况下添加
- 如何确保在使用基于布尔值的两个方法之一调用方法时避免分支预测错误
- 停止cmake target_link_libraries将插件中静态库的两个对象文件链接到静态库本身
- 将fold表达式与std::一起用于两个元组
- 如何在C++中比较两个char数组
- 给定两个偶数,求出它们之间所有偶数的平方和
- 比较两个大小不等的映射c++
- C++需要帮助从用户那里获得一个整数,并确保它在另外两个整数之间
- 如何在for循环中包含两个索引值的测试条件
- 在声明中合并两个常量"std::set"(不是在运行时)
- 如何使用OpenMP并行这两个循环