• 把《c++ primer》读薄(3-2 标准库vector容器+迭代器初探)


    督促读书,总结精华,提炼笔记,抛砖引玉,有不合适的地方,欢迎留言指正。

    标准库vector类型初探,同一种类型的对象的集合(类似数组),是一个类模版而不是数据类型,学名容器,负责管理 和 存储的元素 相关的内存,因为vetcor是类模版,对应多个不同类型,比如int,string,或者自己定义的数据类型等。

    程序开头应如下声明

    #include <iostream>
    #include <vector>
    #include <string>
    using std::string;
    using std::vector;
    using std::cout;
    using std::cin;
    using std::endl;

    简单的vector<xx>类型的变量声明

    vector<int> ivec;//声明一个vector<int>数据类型的变量ivec

    问题1、标准库类型vector初始化的值的类型必须一致!

        vector<int> ivec1;//默认调无参构造
        vector<int> ivec2(ivec1);//直接初始化ivec2为ivec1的一个副本
        vector<string> strvec(ivec2);//error C2514: “std::vector”: 类没有构造函数。说明类型不一致,无法完成初始化!自然报错!

    vector对象的初始化,vector类模版定义了四个构造函数,无参构造函数,直接初始化的带参构造函数,初始化为n个值为i的构造函数还有一种值初始化构造函数。

    下面的也没有问题!

        vector<vector<int>> ivec;//ok,完全没问题!存储的是vector<int>类型的元素

    问题2、勿忘它的两种直接初始化的方式

        vector<int> ivec(20, 10);//初始化ivec为含有20个元素,每个元素=10
        vector<string> svec(10, "hello");//初始化为含义10个元素,每个元是一个字符串hello
    
        //值初始化方式
        vector<int> ivec1(100);//内置类型,比如int类型的元素存在容器vector,那么默认初始化为100个0
        vector<string> svec1(20);//同理若是类类型,如果有默认构造函数,那么按照它的默认构造函数初始化,比如这里是20个空串

    如果既不是带默认构造函数的类类型,也不是c/c++的内置基本类型,那么编程时,需要手动写上初始化值具体是多少。还有一种极端,类类型里没有定义任何的构造函数,那么c++标准库还是会产生一个初始值去依次初始化容器里的元素。

    问题3、需要理解c++标准库容器对象,比如vector容器的一个重要属性!

    标准库容器在运行的同时,可以高效的被添加元素,且不用预先分配内存空间!要知道,vector动态增长的效率高,专家推荐使用,且要知道,这不同于内置基本类型,后续深入,这里要先记住,不需要提前为容器对象分配内存。

    问题4、对vector容器对象的求长度和判空操作

    发现十分类似标准库string对象的操作,还有数组等,很像的。但是肯丢有不同。这里要注意,对于容器vector来说,vector类型总是要说明它包含的元素的类型!不能丢!

        vector<string> svec(10, "null");
        //vector::size_type len = svec.size();//error C2955: “std::vector”: 使用类 模板 需要 模板 参数列
        cout << len << endl;

    改为

        vector<string>::size_type len = svec.size();
        cout << len << endl;//打印10

    判空操作(和标准库类型string的判空类似,空就返回true)

        vector<int> ivec;
        if (ivec.empty())
        {
            cout << ivec.size() << endl;//成功执行,打印0
        }

    问题5、对vector容器添加元素的操作push_back()

        vector<string> svec;
        string str;
        //每次循环把输入的str字符串插入到vector容器对象的后面
        while (cin >> str)
        {
            svec.push_back(str);
        }
        //直到循环结束为止

    问题6、vector容器对象的下标操作(类似string对象的下标)

    可以作为右值,也可以作为左值,同时建议使用标准库容器的size_type类型来定义下标

        vector<string> svec(10, "sss");
        //重置容器内部元素的值为ooxx
        for (vector<string>::size_type i = 0; i != svec.size(); i++)
        {
            svec[i] = "ooxx";
        }

    需要知道的事实:类似size()这样的小型库函数,在c++里都被定义为了内联函数!

    注意:vector容器的下标操作(标准库string类型同样类似),仅仅是只能获取已经存在的元素,不能添加元素!如下是错误的

        vector<int> ivec;
        for (vector<int>::size_type i = 0; i != 100; i++)
        {
            //ivec[i] = i;//这样做是错误的!程序中断!因为ivec是空的vector对象!下标操作只能针对已经存在的元素
            ivec.push_back(i);//这样就对了,从尾部插入
        }

    且要知道,vector容器的下标也是类似string对象或者数组,从0开始

        vector<int> ivec(10, 1);
        //int i = ivec[10];//产生运行时错误,程序中断,不存在元素下标为10的

    这样的错误,就是常见的缓冲区溢出错误,很常见,需要注意,对数组也适用,还有标准库string类型

    问题7、输入一组整数到vector对象,相邻的元素相加输出,并提示奇数的情况。

    错误1:goto语句造成死循环

        int in;
        vector<int> ivec;
    
    begin:
        cout << "请输入一组整数:注意ctrl+z结束输入!" << endl;
    
        while (cin >> in)
        {
            ivec.push_back(in);
        }
        //判断空输入否
        if (0 == ivec.size())
        {
            cout << "输入为空,重新输入!" << endl;
            goto begin;
        }

    错误2:下标溢出错误

    for (vector<int>::size_type i = 0; i < ivec.size(); i = i + 2)
        {
            cout << ivec[i] + ivec[i + 1] << "	";
            //每行输出的个数控制为5个。必须i+1,因为i初始值=0
            if (0 == (i + 1) % 5)
            {
                cout << endl;
            }
        }

    i<ivec.size()这里出错,下标溢出,只有元素个数是偶数的时候不错,奇数就溢出了。修改后:

     1 #include <iostream>
     2 #include <vector>
     3 #include <string>
     4 using std::string;
     5 using std::vector;
     6 using std::cout;
     7 using std::cin;
     8 using std::endl;
     9 
    10 int main(void)
    11 {
    12     int in;
    13     vector<int> ivec;
    14     cout << "请输入一组整数:注意ctrl+z结束输入!" << endl;
    15 
    16     while (cin >> in)
    17     {
    18         ivec.push_back(in);
    19     }
    20     //判断空输入否
    21     if (0 == ivec.size())
    22     {
    23         cout << "输入为空,重新输入!" << endl;
    24         system("pause");
    25         return -1;
    26     }
    27     //求相邻的元素的和,关键算法!
    28     //因为是相邻元素,故判断完毕,i+2之后再赋给i,跨度为2
    29     //且不知道输入的元素是奇数,肯丢留最后一个元素不计算,是偶数则正好计算完毕!
    30     //这里必须是i < size -1,否则下标溢出错误,偶数没问题,主要是奇数的话,如果到倒数第二个元素还+2,必然是溢出错误!
    31     for (vector<int>::size_type i = 0; i < ivec.size() - 1; i = i + 2)
    32     {
    33         cout << ivec[i] + ivec[i + 1] << "	";
    34         //每行输出的个数控制为5个。必须i+1,因为i初始值=0
    35         if (0 == (i + 1) % 5)
    36         {
    37             cout << endl;
    38         }
    39     }
    40     //判断奇偶
    41     if (0 != ivec.size() % 2)
    42     {
    43         cout << "元素个数为奇数,最后一个元素" << ivec[ivec.size() - 1] << "被忽略!" << endl;
    44     }
    45     
    46     cout << endl;
    47 
    48     system("pause");
    49     return 0;
    50 }

    问题8、读入一组整数到vector对象,使得头尾元素两两配对,计算和,并输出奇数元素个数的提示。

     1     vector<int> ivec;
     2     int in;
     3     cout << "输入一组正数,ctrl+z结束输入:" << endl;
     4 
     5     while (cin >> in)
     6     {
     7         ivec.push_back(in);
     8     }
     9 
    10     if (0 == ivec.size())
    11     {
    12         cout << "空容器,必须输入元素!" << endl;
    13         system("pause");
    14         return -1;
    15     }
    16     //首末元素相加的处理,具有模版性质,也就是设计程序的通用性
    17     //如果是奇数元素,那么中间的会留下,如果是偶数元素,没这个问题
    18     vector<int>::size_type first = 0;
    19     vector<int>::size_type last = 0;
    20     vector<int>::size_type count = 0;//计数,控制打印输出
    21 
    22     for (first = 0, last = ivec.size() - 1; first < last; first++, last--)
    23     {
    24         cout << ivec[first] + ivec[last] << "	";
    25         count++;
    26         //控制打印输出每行3个数
    27         if (0 == count % 3)
    28         {
    29             cout << endl;
    30         }
    31     }
    32     //判断奇数还是偶数,给出提示
    33     //灵活!使用简单的方式,如果是奇数的话,必然最后last和first重合
    34     if (first == last)
    35     {
    36         cout << "中间的元素" << ivec[first] <<"留下了,因为元素个数是奇数!" << endl;
    37     }
    38 
    39     cout << endl;

    问题9、读入一段文本到vector对象,把对象中每个元素里面的单词都转换为大写之后在输出,5个一行。

     1 #include <iostream>
     2 #include <vector>
     3 #include <string>
     4 #include <cctype>
     5 using std::string;
     6 using std::vector;
     7 using std::cout;
     8 using std::cin;
     9 using std::endl;
    10 
    11 int main(void)
    12 {
    13     vector<string> svec;
    14     string str;
    15 
    16     while (cin >> str)
    17     {
    18         svec.push_back(str);
    19     }
    20 
    21     if (0 == svec.size())
    22     {
    23         return -1;
    24     }
    25 
    26     for (vector<string>::size_type i = 0; i != svec.size(); i++)
    27     {
    28         for (string::size_type j = 0; j != svec[i].size(); j++)
    29         {
    30             //如果是小写字母
    31             if (islower(svec[i][j]))
    32             {
    33                 //转换为大写输出
    34                 svec[i][j] = toupper(svec[i][j]);
    35             }
    36         }
    37 
    38         cout << svec[i] << "	";
    39 
    40         if (0 == (i + 1) % 5)
    41         {
    42             cout << endl;
    43         }
    44     }
    45 
    46     system("pause");
    47     return 0;
    48 }

    问题10、迭代器入门

    //c++为每一种标准容器都定义了一种迭代器类型,迭代器是一种——可以检测容器内的元素并且可以遍历元素的数据类型。除了使用下标来访问vector对象之外,还可以使用迭代器访问,比下标操作更方便,更通用,因为所有的标准库的容器都支持迭代器,但是只有部分容器支持下标操作,故成熟的c++程序员应该使用迭代器,而不是下标操作访问容器内的元素。

    //vector容器的迭代器类型
        vector<int>::iterator iter;//定义一个iter变量,它的数据类型是vector<int>定义的迭代器类型

    //记住,每个标准库容器都定义了自己的迭代器类型,用来遍历自己容器内的元素。

        //每个容器都定义了begin和end函数,目的是返回迭代器,如果容器不空,则begin返回的迭代器指向容器内的第一个元素
        vector<int> ivec;
        vector<int>::iterator iter;
        iter = ivec.begin();//iter变量被初始化,使用容器的begin函数返回的迭代器,此时iter指该元素为ivec[0]

    //恰恰相反,end函数返回的迭代器,指向容器内末端元素的下一个元素!记住是下一个!不是最后一个!故end操作返回的也叫超出末端迭代器。说明end函数返回的迭代器指向一个不存在的元素,不指向容器内任何实际存在的元素!作用是哨兵!表示我们已经处理完毕容器所有元素!

    如果容器为空,则begin函数返回的迭代器和end函数返回的迭代器相同。

    问题11、vector迭代器的自增、解引用、和比较相等否的操作

    如果想要获取迭代器指向的元素的值,可以使用类似指针的操作,解引用操作!*iter就代表迭代器iter指向的元素的值!比如iter迭代器指向的元素是容器内第一个ivec[0],那么*iter和ivec[0]是等价语句!

    如果想让迭代器类似下标那样,移动自己的指向,则可以使用迭代器的自增操作。比如iter++就是迭代器向前移动一个元素的位置!除了这些,迭代器也可以进行比较操作,==、!=操作来比较容器的迭代器,如果两个迭代器指向同一个元素,则==为真,否则为假。

    注意,end函数不指向容器内的元素,故不能对它使用自增或者解引用操作!

        //新的赋值方式
        vector<int> ivec(10, 2);
        //把容器ivec的元素重置为0
    
        //for (vector<int>::size_type i = 0; i != ivec.size(); i++)
        //{
        //    ivec[i] = 0;//这是old方法
        //}
    
        //比较经典常用的方法如下:
        for (vector<int>::iterator iter = ivec.begin(); iter != ivec.end(); iter++)
        {
            *iter = 0;
        }

    如果容器为空,则begin函数和end函数返回的迭代器相等,for循环测试失败,没问题!

    问题12、两类只读迭代器类型

    //类似普通的const常量,但是有区别。比如,如果仅仅想遍历容器的元素,对迭代器有const_iterator类型的迭代器,对它解引用,得到的是指向const对象的引用

     1     string str;
     2     vector<string> svec;
     3     
     4     while (cin >> str)
     5     {
     6         svec.push_back(str);
     7     }
     8 
     9     for (vector<string>::const_iterator iter = svec.begin(); iter != svec.end(); iter++)
    10     {
    11         //*iter = "dada";//error,const_iterator类型的迭代器,本身可以被改变,比如自增,但是迭代器指向的容器内的元素不能被修改!
    12         //和普通的const常量有一些区别!有些类似指向常量的指针,指针本身可以变,但是指向的内容不能修改。
    13     }

    来对比const类型的迭代器,类似常指针,本身定义的时候必须初始化,本身不能被修改,但是指向的内容可以修改,如

    1     vector<int> ivec(10);
    2     //const类型的iter必须初始化
    3     const vector<int>::iterator iter = ivec.begin();
    4     //初始化之后不能被修改
    5     //iter++;//error
    6     //但是iter指向的内容可以被修改,对比,const_iterator类型的迭代器,类似指向常量的指针,迭代器本身能修改,指向的元素不能修改,,和他/她相反
    7     *iter = 1;

    注意区分两者,不要混淆。总结:

    const 迭代器是迭代器常量

    该迭代器本身的值不能修改,即该迭代器在定义时需要初始化,而且初始化之后,不能再指向其他元素。若需要指向固定元素的迭代器,则可以使用const 迭代器。但是它指向的元素的值可以被修改!

    const_iterator 是一种迭代器类型

    对这种类型的迭代器解引用会得到一个指向const 对象的引用,即通过这种迭代器访问到的对象是常量。该被指向的对象不能修改,因此,const_iterator 类型只能用于读取容器内的元素,不能修改元素的值。若只需遍历容器中的元素而无需修改它们,则可以使用const_iterator。但是迭代器本身能被修改。两者相反!

    问题13、迭代器的算术操作

    其他容器的迭代器类似。除了自增、自减之外,还有其他算术运算适用。

     1     vector<int> ivec(10);
     2     //const类型的iter必须初始化
     3     vector<int>::iterator iter = ivec.begin();
     4     //iter++;//ok
     5     //iter--;//ok
     6 
     7     //iter + 5;//ok,对一个迭代器对象加一个整型值,使得iter指向新的元素第6个元素,注意不要越界
     8     //但是,加上或者减去的整型,最好是size_type类型的!
     9 
    10     //iter - 100;//error,越界,程序中断!
    11     
    12     //iter + 10;//ok,加的时候,可以加到最后一个元素的下一位。
    13 
    14     //还能求两个迭代器的距离
    15     vector<int>::iterator iter1 = ivec.end();
    16 
    17     cout << iter1 - iter << endl;//打印10
    18 
    19     //注意,这里相减得到的值,可能是负数,也可以是正数
    20     cout << iter - iter1 << endl;//打印-10
    21     //这就说明,这个值的类型不再是容器的size_type类型,而是新的容器的类型,叫:
    22     //differenec_type
    23     vector<int>::difference_type i = iter - iter1;//ok,是带符号类型

    注意,迭代器是没有相加操作的!比如:

        iter1 + iter1;//报错!

    也就是说,要求得容器的中间元素,不能这样写:

    vector<int>::iterator mid = (vi.begin() + vi.end())/2;

    但是可以这样:

    vector<int>::iterator mid = vi.begin() + vi.end()/2;

    直接定位容器的中间元素,简单高效,不再需要一次次的去自增或者自减的遍历了。

    欢迎关注

    dashuai的博客是终身学习践行者,大厂程序员,且专注于工作经验、学习笔记的分享和日常吐槽,包括但不限于互联网行业,附带分享一些PDF电子书,资料,帮忙内推,欢迎拍砖!

  • 相关阅读:
    发现对各类项目有用的不同JavaScript的Web UI
    PowerDesigner 15.1 安装步骤详细图解及破解
    数据库设计工具PowerDesigner基础普及
    Vistual Studio 2010(VS2010)安装 MVC3.0具体方法
    pb的网络资源【转】
    powerbuider11 C/S 转换为B/S
    转:将可执行文件注册成系统windows服务
    WCF绑定类型选择(转)
    (转)找增强方法总结
    ALV简单模板1
  • 原文地址:https://www.cnblogs.com/kubixuesheng/p/4138585.html
Copyright © 2020-2023  润新知