如果说std::copy是STL中最简单的算法,那么std::transform就是第二简单的算法。和copy类似,其可将容器某一范围的元素放置到其他容器中,在这个过程中允许进行一些变换(变换函数会对输入值进行一定操作,然后再赋给目标容器)。此外,两个具有不同元素类型的容间也可以使用这个函数。这个函数超级简单,并且非常有用,这个函数会让标准组件具有更好的可移植性。
本节,我们将使用std::transform在拷贝的同时,修改vector中的元素:
-
包含必要的头文件,并且声明所使用的命名空间:
#include <iostream> #include <vector> #include <string> #include <sstream> #include <algorithm> #include <iterator> using namespace std;
-
vector由简单的整数组成:
int main() { vector<int> v {1, 2, 3, 4, 5};
-
现在,我们为了打印元素,会将所有元拷贝素到ostream_iterator适配器中。transform函数可以接受一个函数对象,其能在拷贝过程中对每个元素进行操作。这个例子中,我们将计算每个元素的平方值,所以代码将打印出平方数,并且因为直接进行了打印,平方数并没有进行保存:
transform(begin(v), end(v), ostream_iterator<int>{cout, ", "}, [] (int i) { return i * i; }); cout << '\n';
-
让我来做另一个变换。例如,对于数字3来说,显示成
3^2 = 9
显然有更好的可读性。下面的辅助函数int_to_string函数对象就会使用std::stringstream对象进行打印操作:auto int_to_string ([](int i) { stringstream ss; ss << i << "^2 = " << i * i; return ss.str(); });
-
这样就可以将整型值放入字符串中。可以说我么将这个证书映射到字符串中。使用transform函数,使我们可以拷贝所有数值到一个字符串vector中:
vector<string> vs; transform(begin(v), end(v), back_inserter(vs), int_to_string);
-
在打印完成后,我们的例子就结束了:
copy(begin(vs), end(vs), ostream_iterator<string>{cout, "\n"}); }
-
编译并运行程序:
$ ./transforming_items_in_containers 1, 4, 9, 16, 25, 1^2 = 1 2^2 = 4 3^2 = 9 4^2 = 16 5^2 = 25
std::transform函数工作原理和std::copy差不多,不过在拷贝的过程中会对源容器中的元素进行变换,这个变换函数由用户提供。