C++迭代器注意事项
iterator优先于其它迭代器
iterator优先于const_iterator,reverse_iterator,const_reverse_iterator,因为基本上大多数容器函数都是以iterator为参数,不支持其它三种迭代器,iterator转为const_iterator比较方便,但是反过来转换编译器就会报错,即使使用const_cast也无济于事,所以尽量还是优先使用iterator,如果真的得到了一个const_iterator,函数却需要iterator,可以使用advance和distance搭配来"转换".
using IntDeque = deque<int>;
using Iter = IntDeque::iterator;~~~~
using ConstIter = IntDeque::const_iterator;
IntDeque d;
ConstIter ci;
Iter i(d.begin());
advance(i, distance<ConstIter>(i, ci));
distance用于取得两个指向同一个容器的迭代器之间的距离
advance用于将一个迭代器移动指定的距离
注意这里distance使用了模板,需要显式指明参数类型,将两个迭代器统一看成了一种迭代器,如果指明类型,以上将是两种迭代器,模板推导出错,编译失败。
至于这种转换的效率如何,取决于迭代器是随机访问迭代器还是其它双向迭代器等。
reverse_iterator转换问题
reverse_iterator转成iterator的一些问题,需要明白转为iterator的目的,用于插入操作还是用于删除操作,用于插入操作比较方便,直接调用base()函数转换再插入即可,这里涉及到转换后位置下标对应的问题,需要画图说明(不会markdown画图),网上搜base()位置的问题即可,如果涉及删除操作需要将迭代器首先递增一次再转换删除.
vector<int> v;
vector<int>::reverse_iterator ri;
v.erase((++ri).base());
istreambuf_iterator vs istream_iterator
对于逐个字符的输入考虑使用istreambuf_iterator,该迭代器是直接从流的缓冲区中读去下一个字符,速度快,istream_iterator类似于>>操作符,这会忽略空格字符,而且这个迭代器会考虑很多,会检查很多异常情况,如果只是想从输入流中读取一些字符,大材小用而且效率低,如果想要禁止忽略空格可以设置
ifstream file("xxx.txt");
file.unsetf(ios::skipws);
string fileData((istream_iterator<char>(file)), istream_iterator<char>());
而直接使用istreambuf_iterator因为直接从缓冲区里读,不会忽略任何字符
ifstream file("xxx.txt");
string fileData((istreambuf_iterator<char>(file)), istreambuf_iterator<char>());