我有一个声明如下的地图:
map < string, set<string> > index;
std::map < string, set<string> >::iterator iter;
我想打印index
的内容。因此,我写了以下内容:
for (iter = index.begin(); iter!= index.end(); iter++){
cout << iter->first << endl;} // It's OK.
但是如果我尝试打印iter->second
,上面的代码不起作用:
for (iter = index.begin(); iter!= index.end(); iter++){
cout << iter-> second << endl;} // Doesn't work.
我想这是因为second
中的index
是一组字符串而不是简单的字符串。但是我该如何打印呢?
如果你想打印
iter->second
使用operator <<,您必须覆盖此运算符!否则,使用set迭代器迭代iter-> second rappresent的字符串集:
set<string>::iterator iterator_;
迭代很容易,忍者解决方案是重载运算符! :P
迭代:
map<string, set<string> >::iterator iter_;
for (iter_ = index_.begin(); iter_ != index_.end(); iter_++) {
cout << iter_->first << " - ";
set<string>::iterator iterator_;
for (iterator_ = s.begin(); iterator_ != s.end(); iterator_++) {
cout << *iterator_ << " - ";
}
cout << endl;
}
class MySet: public set<string> {
friend ostream& operator<<(ostream& out, const MySet& s) {
set<string>::iterator iterator_;
for (iterator_ = s.begin(); iterator_ != s.end(); iterator_++) {
out << *iterator_;
}
return out;
}
void test() {
map<string, MySet> index_;
map<string, MySet>::iterator iter_;
for (iter_ = index_.begin(); iter_ != index_.end(); iter_++) {
cout << iter_->first << " - ";
cout << iter_->second;
}
}
};
我在过去几年中使用了以下功能。如果重载operator <<,你必须要小心,因为你最终可能会给编译器带来错误,因为它无法在标准operator <<和你的自定义operator <<之间进行选择。
随着C ++ 11引入lambdas和基于范围的for循环,我可能不太经常需要它。
#include <vector>
#include <iostream>
#include <algorithm>
#include <iterator>
using namespace std;
template<typename C>
void print_container(ostream& os, const C& c)
{
copy(c.begin(), c.end(), ostream_iterator<int>(os, " "));
}
int main(void)
{
vector<int> v;
v.push_back(1);
v.push_back(2);
v.push_back(3);
print_container(v);
}
(在我看来)C ++中最令人讨厌的事情是输出运算符没有为标准库类型定义。
你需要一些代码有点像:
typedef std::map < string, set<string> >::iterator iter;
for (iter = index.begin(); iter!= index.end(); iter++) {
std::cout << iter->first << " : {";
for(set<string>::iterator it = iter->second.begin(); it != iter->second.end(); ++it) {
std::cout << *it << " ";
}
std::cout << "}\n";
}
正如您所说,iter->second
代表代码中的一组字符串。你不能简单地打印它,因为输出流的operator<<
默认不知道如何格式化std::set
。正如Luca Davanzo的answer所解释的那样,一种解决方案是为你的装置重载该操作符。
另一个解决方案,如Chris Jefferson的answer所示,就是简单地添加另一个循环来迭代你的集合。但是,通过使用以下C++11功能可以提高该解决方案的可读性:range-based for loops和占位符auto。因此,您的代码可能看起来很短:
std::map<std::string, std::set<std::string>> index;
index["a"] = { "v", "w" };
index["b"] = { "x", "y", "z" };
for (const auto &mapItem : index) {
std::cout << "m[" << mapItem.first << "] =";
for (const auto &setItem : mapItem.second)
std::cout << " " << setItem;
std::cout << std::endl;
}
输出:
m [a] = v w m [b] = x y z
从C++17开始,你可以使用基于范围的for循环和structured bindings。这样可以进一步提高可读性,因为您可以从代码中删除first
和second
成员:
for (const auto &[k, v] : index) {
std::cout << "m[" << k << "] =";
for (const auto &s : v)
std::cout << " " << s;
std::cout << std::endl;
}