随着C++11标准的出现,C++标准添加了许多有用的特性,C++代码的写法也有比较多的变化。vector是经常要使用到的std组件,对于vector的遍历,本文罗列了若干种写法。(注:本文中代码为C++11标准的代码,需要在较新的编译器中编译运行)
假设有这样一个vector:(注意,这种列表初始化的方法是c++11中新增语法)
vector<int> valList = { 0, 1, 2, 3, 4, 5, 6, 7, 8, 9 };
需要输出这个vector中的每个元素,测试原型如下:
void ShowVec(const vector<int>& valList)
{
}
int main(int argc, char* argv[])
{
vector<int> valList = { 0, 1, 2, 3, 4, 5, 6, 7, 8, 9 };
ShowVec(valList);
return 0;
}
下面就试试几种写法吧!
方法零:对C念念不舍的童鞋们习惯的写法
void ShowVec(const vector<int>& valList)
{
int count = valList.size();
for (int i = 0; i < count;i++)
{
cout << valList[i] << endl;
//或者 cout << valList.at(i) << endl;
}
}
方法一:大家喜闻乐见的for循环迭代器输出,(注意,此处使用了C++11中新增的标准库容器的cbegin函数)
void ShowVec(const vector<int>& valList)
{
for (vector<int>::const_iterator iter = valList.cbegin(); iter != valList.cend(); iter++)
{
cout << (*iter) << endl;
}
}
或者使用c++新增的语义auto,与上面差不多,不过能少打几个字:
void ShowVec(const vector<int>& valList)
{
for (auto iter = valList.cbegin(); iter != valList.cend(); iter++)
{
cout << (*iter) << endl;
}
}
方法二:for_each加函数
template<typename T>
void printer(const T& val)
{
cout << val << endl;
}
void ShowVec(const vector<int>& valList)
{
for_each(valList.cbegin(), valList.cend(), printer<int>);
}
方法三:for_each加仿函数:
template<typename T>
struct functor
{
void operator()(const T& obj)
{
cout << obj << endl;
}
};
void ShowVec(const vector<int>& valList)
{
for_each(valList.cbegin(), valList.cend(), functor<int>());
}
方法四:for_each加Lambda函数:(注意:lambda为c++11中新增的语义,实则是一个匿名函数)
void ShowVec(const vector<int>& valList)
{
for_each(valList.cbegin(), valList.cend(), [](const int& val)->void{cout << val << endl; });
}
方法五:for区间遍历:(注意,for区间遍历是c++11新增的语法,用于迭代遍历数据列表)
for (auto val : valList)
{
cout << val << endl;
}
加几个与本文无太大关系知识点:
一、vector报错原因:
(1)当我们使用memset初始化Base类时,其实也把所有的这些指针都置0了。所以,使用memset的时候还是需要很慎重,至少有stl容器的时候最好不要用这种方式去初始化
(2) 在erase操作后,没有将循环变量i指向修改后的向量迭代器,就继续循环,再与end()比较时断言出现。
for (vector<int>::iterator i= vector.begin();i != vector.end();i++) // 这里报错
{
vector.erase(i);
....
}
解决方法是将“vector.erase(i);”替换为“i = vector.erase(i);”,这是因为STL里的所有容器类中的erase实现都会返回一个迭代器,这个迭代器指向了“当前删除元素的后继元素,或是end()”。
(3)类型不匹配,例如用int型的向量迭代器与char型的向量迭代器进行比对操作
二、获取程序最后一次报错原因
void ErrorExit(LPCWSTR lpszFunction) //传入参数为函数指针
{
TCHAR szBuf[80];
LPVOID lpMsgBuf;
DWORD dw = GetLastError(); //错误代号
FormatMessage(
FORMAT_MESSAGE_ALLOCATE_BUFFER |
FORMAT_MESSAGE_FROM_SYSTEM,
NULL,
dw,
MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT),
(LPTSTR) &lpMsgBuf,
0, NULL);
wsprintf(szBuf, L"%s failed with error %d: %s", lpszFunction, dw, lpMsgBuf);
MessageBox(NULL, szBuf, L"Error", MB_OK);
LocalFree(lpMsgBuf);
ExitProcess(dw);
}