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); 
}

三、map和vector详解