这是我提出的一种可能的方法:
struct RetrieveKey
{
template <typename T>
typename T::first_type operator()(T keyValuePair) const
{
return keyValuePair.first;
}
};
map<int, int> m;
vector<int> keys;
// Retrieve all keys
transform(m.begin(), m.end(), back_inserter(keys), RetrieveKey());
// Dump all keys
copy(keys.begin(), keys.end(), ostream_iterator<int>(cout, "\n"));
当然,我们也可以通过定义另一个函数RetrieveValues从映射中检索所有值。
有没有其他方法可以轻松实现这个目标?(我总是想知道为什么std::map不包括我们这样做的成员函数。)
在Eric Niebler的range-v3库中,你可以使用ranges::to将一个range直接写入到容器中(希望很快在std中,也许c++ 26?):
(演示)
#include <fmt/ranges.h>
#include <map>
#include <range/v3/all.hpp>
int main() {
std::map<int, int> m{ {1, 100}, {2, 200}, {3, 300} };
auto keys{ m | ranges::views::keys | ranges::to<std::vector<int>>() };
fmt::print("{}", keys);
}
// Outputs: [1, 2, 3]
通过c++ 17的结构化绑定(“解构”)声明语法,
你可以这样做,这样更容易理解。
// To get the keys
std::map<int, double> map;
std::vector<int> keys;
keys.reserve(map.size());
for(const auto& [key, value] : map) {
keys.push_back(key);
}
// To get the values
std::map<int, double> map;
std::vector<double> values;
values.reserve(map.size());
for(const auto& [key, value] : map) {
values.push_back(value);
}
我认为上面提出的BOOST_FOREACH是漂亮和干净的,然而,还有另一个使用BOOST的选项。
#include <boost/lambda/lambda.hpp>
#include <boost/lambda/bind.hpp>
std::map<int, int> m;
std::vector<int> keys;
using namespace boost::lambda;
transform( m.begin(),
m.end(),
back_inserter(keys),
bind( &std::map<int,int>::value_type::first, _1 )
);
copy( keys.begin(), keys.end(), std::ostream_iterator<int>(std::cout, "\n") );
就我个人而言,我不认为这种方法在这种情况下像BOOST_FOREACH方法那样干净,但是boost::lambda在其他情况下可以非常干净。