我有以下测试程序:
#include <vector>
#include <print>
#include <ranges>
int main() {
const std::vector<int> input = {0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10};
auto output = input
| std::views::filter([](const int n) { std::print("{} ", n); return n % 3 == 0; })
| std::views::transform([](const int n) {return n * n; });
const std::vector<int> vector = std::ranges::to<std::vector>( output );
std::println("\ninput size: {} output size: {}", input.size(), vector.size() );
}
在 Xcode 16.2 中使用 c++23 模式进行编译时,输出以下内容:
0 1 2 3 4 5 6 7 8 9 10 1 2 3 4 5 6 7 8 9 10
input size: 11 output size: 4
我对此感到困惑。为什么在转换为 std::vector 时每个项目都要评估两次?
当我尝试使用一个简单的for
循环来打印元素时,每个项目仅被评估一次:
#include <vector>
#include <print>
#include <ranges>
int main() {
const std::vector<int> input = {0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10};
auto output = input
| std::views::filter([](const int n) { std::print("{} ", n); return n % 3 == 0; })
| std::views::transform([](const int n) {return n * n; });
int sum = 0;
for ( int i : output ) sum += i;
std::println("\nsum: {}", sum);
}
输出:
0 1 2 3 4 5 6 7 8 9 10
sum: 126
第一个程序似乎效率低下,让我重新考虑是否应该开始使用范围......有没有更好的方法来编写它?
std::ranges::to<std::vector>( output )
在初始化元素之前计算的长度output
。也就是说,有两次迭代:output
(也是结果的大小std::vector
)。std::vector
。这种方法通常比单次迭代更有效,因为它避免了在向量构造期间进行重新分配。