C++面向对象(OOP)编程-STL详解(vector)

本文主要介绍STL六大组件,并主要介绍一些容器的使用。

目录

1 泛型编程

2 C++STL

3 STL 六大组件

4 容器

4.1 顺序性容器

4.1.1 顺序性容器的使用场景

4.2 关联式容器

4.2.1 关联式容器的使用场景

4.3 容器适配器

4.3.1 容器适配器的使用场景

5 具体容器的使用和剖析

5.1 vector(向量)

5.1.2 vector扩容


1 泛型编程

        泛型编程是一种代码重用技术,尽可能的将代码写的抽象和通用,将算法从数据结构抽象出来,以便适配多种多样的数据结构,C++的模板编程就是一种泛型编程技术。

2 C++STL

        STL(Standard Template Library),即标准模板库,是一个高效的C++程序库。

    被容纳于C++标准程序库(C++ Standard Library)中,是ANSI/ISO C++标准中最新的也是极具革命性的一部分。

    包含了诸多在计算机科学领域里常用的基本数据结构和基本算法。为广大C++程序员们提供了一个可扩展的应用框架,高度体现了软件的可复用性。

        从逻辑层次来看,在STL中体现了泛型化程序设计的思想(generic programming)

     在这种思想里,大部分基本算法被抽象,被泛化,独立于与之对应的数据结构,用于以相同或相近的方式处理各种不同情形。

        从实现层次看,整个STL是以一种类型参数化(type parameterized)的方式实现的,基于模板(template)。

        简单理解:

        STL 的基本观念就是将数据和操作分离。数据由容器进行管理,操作则由算法进行,而迭代器在两者之间充当粘合剂,使任何算法都可以和任何容器交互运作。

3 STL 六大组件

STL 六大组件
STL的组成含义
容器一些封装数据结构的模板类,例如 vector 向量容器、list 列表容器等。
算法STL 提供了非常多(大约 100 个)的数据结构算法,它们都被设计成一个个的模板函数,这些算法在 std 命名空间中定义,其中大部分算法都包含在头文件 <algorithm> 中,少部分位于头文件 <numeric> 中。
迭代器在 C++ STL 中,对容器中数据的读和写,是通过迭代器完成的,扮演着容器和算法之间的胶合剂。
函数对象如果一个类将 () 运算符重载为成员函数,这个类就称为函数对象类,这个类的对象就是函数对象(又称仿函数)。
适配器可以使一个类的接口(模板的参数)适配成用户指定的形式,从而让原本不能在一起工作的两个类工作在一起。值得一提的是,容器、迭代器和函数都有适配器。
内存分配器为容器类模板提供自定义的内存申请和释放功能,由于往往只有高级用户才有改变内存分配策略的需求,因此内存分配器对于一般用户来说,并不常用。简称分配器。

4 容器

        所谓容器,就是可以承载,包含元素的一个器件,它是STL六大组件之一,是容器、算法、迭代器中最重要也是最核心的一部分。

4.1 顺序性容器

        顺序性容器就是将一组具有相同类型的元素以严格的线性形式组织起来。顺序性容器的存储结构有顺序存储和链式存储。

具体的顺序性容器如下:

容器

简介说明

vector

可变大小数组。相当于数组,可动态构建,支持随机访问,无头插和尾插,仅支持inset插入,除尾部外的元素删除比较麻烦。但使用最为广泛。

deque

双端队列。支持头插、删,尾插、删,随机访问较vector容器来说慢,但对于首尾的数据操作比较方便

list

双向循环链表。使用起来很高效,对于任意位置的插入和删除都很快,在操作过后,以后指针、迭代器、引用都不会失效

forward_list

单向链表。只支持单向访问,在链表的任何位置进行插入/删除操作都非常快

array

固定数组。vector的底层即为array数组,它保存了一个以严格顺序排列的特定数量的元素

4.1.1 顺序性容器的使用场景

        一般大多数的题目都可以使用vector容器,除非有特定需求使用其他容器更加合理方便;

如果需要在一串数字的头尾进行操作,偏向deque,对于较中间的元素操作,不推荐;

        对于中间的元素插入或删除,可采用forward_list(单向链表)或list(双向链表),不需要移动元素,只需改变相关结点的指针域即可;

一个例子:

#include <iostream>
#include <vector>

using namespace std;

// vector容器大小:
// 1 2 4 8 16 32
// vector 容器大小的增长是以2的倍数
int main(int argc, char *argv[])
{

    vector<int> v1;

    for (int i = 0;i < 17;i++)
        v1.push_back(i);

    cout << v1[3] << endl;
    cout << v1.size() << endl;
    cout << v1.capacity() << endl;


    return 0;
}

        运行结果:

4.2 关联式容器

        关联式容器每一个元素都有一个键值(key),对于二元关联容器,还拥有实值(value)容器中的元素顺序不能由程序员来决定,有set(集合)和map(映射)这两大类,它们均是以RB-Tree(red-black tree,红黑树)为底层架构。

具体的关联式容器如下:

容器

简介说明

set/mutliset

集合/多重集合。对于set,在使用insert插入元素时,已插入过的元素不可重复插入,这正好符合了集合的互异性,在插入完成显示后,会默认按照升序进行排序,对于multiset,可插入多个重复的元素

map/mutlimap

映射/多重映射。二者均为二元关联容器(在构造时需要写两个参数类型,前者对key值,后者对应value值),因为有两个参数,因此在插入元素的时候需要配合对组pair进行插入,具体见深入详解

4.2.1 关联式容器的使用场景

        如果只负责查找内容,具体到某个单位,使用场景比如对手机游戏的个人的计分的存储,可以使用set或mutliset。

        如果需要同时放入容器的数据不止一个,并且是不同类型,比如一个为整型int,一个为string字符串型,就可以考虑使用map或mutlimap。

4.3 容器适配器

        容器适配器是一个封装了序列容器的一个类模板=,它在一般的序列容器的基础上提供了一些不同的功能。之所以称为容器适配器,是因为它是适配容器来提供其它不一样的功能。通过对应的容器和成员函数来实现我们需要的功能。

具体的容器适配器如下:

容器

简介说明

stack

堆栈。其原理是先进后出(FILO),其底层容器可以是任何标准的容器适配器,默认为deque双端队列

queue

队列。其原理是先进先出(FIFO),只有队头和队尾可以被访问,故不可有遍历行为,默认也为deque双端队列

pirority_queue

优先队列。它的第一个元素总是它所包含的元素中优先级最高的,就像数据结构里的堆,会默认形成大堆,还可以使用仿函数来控制生成大根堆还是生成小根堆,若没定义,默认使用vector容器

4.3.1 容器适配器的使用场景

        (1)对于 stack 堆栈,在我们日常生活中类似于坐地铁、电梯;

        (2)对于 deque 队列,在我们日常生活中类似于排队打饭;

        (3)对于 pirority_queue,因为其本质是堆,可以考虑解决一些贪心问题;

5 具体容器的使用和剖析

5.1 vector(向量)

        对于vector容器,它的数据结构与数组非常类似,但是他们之间的不同之处是数组是静态空间,一旦配置了就不能更改,vector却可以进行动态分配,随着元素的插入和删除,内部的空间也会灵活变动,就和C语言中的malloc和C++中的new是一个道理,不用害怕空间不足而一开始就定义一个很大的数组,节省了内存空间。容器的大小是可以改变的。vector扩容是2的倍数。

一些例子:

#include <iostream>
#include <vector>

/*
 * 线性表是一种逻辑结构,按照存储结构可以分为顺序表和链表
 *
 * 
*/
/* vector 本质上是一个动态变长数组,顺序表,连续的存储空间,访问的时间复杂度为O(1),对于尾部元素的插入和删除时间复杂度都是常量级别的
 * vector 也是一个类模板,vector底层本质就是一个顺序表,它是一个可变长的数组,采用连续存储的空间来存储数据,它的元素类型也可以是任意的内置类型或者自定义类型。
 * 对于vector的扩容机制,Linux一般是以2的倍数增加,VS一般是以1.5的倍数增加,增加快的性能会比较好,但是对空间的浪费会增大;
 * vector扩容是开辟一段新的空间,将旧的数据拷贝到新的空间
*/
/*
 * 扩容 vec.resize(n)  vec.reserve(n)
*/

int main(int argc, char *argv[])
{
    
    std::vector<int> vec = {1, 2, 3, 4, 5};
    // vec.begin()+2 代表从第三个元素开始,vec。begin()代表从第一个元素开始
    std::vector<int> vec1(vec.begin()+2,vec.end()); // = vec
    std::vector<int> vec2(vec); // = vec
    std::vector<int> vec3(4); // [0,0,0,0]
    std::vector<int> vec4(2,4); // [4,4]

    // vec.erase(vec.begin()+1); // 删除第二个元素

    vec.erase(vec.begin(),vec.begin()+1);//删除[1,3) 删除两个元素


    for (auto i : vec1) {
        std::cout << i << " ";
        std::cout << "*******" << std::endl;
    }
    vec.push_back(18); //在尾部插入一个元素
    std::cout << "Front1: " << vec.front() << std::endl;
    std::cout << "Back1: " << vec.back() << std::endl;
    vec.pop_back(); // 弹出尾部的元素
    vec.insert(vec.begin()+4,3,99);
    for (int i = 0;i < vec.size();i++)
    {
        // std::cout << "vec(" << i << "): " << vec[i] << std::endl;
        std::cout << "vec(" << i << "): " << vec.at(i) << std::endl;
        
    }

    for (int i = 0;i < vec.size();i++)
    {
        std::cout << i << " : " << vec.data()[i] << std::endl;
    }
    std::cout << "*********" << std::endl;

    std::vector<int>::iterator it;
    for (it = vec.begin();it != vec.end();it++)
    {
        std::cout << " " << " : " << *it << std::endl;
    }
    std::cout << "*********" << std::endl;

    for (auto it = vec.begin();it != vec.end();it++)
    {
        std::cout << " " << " : " << *it << std::endl;
    }
    std::cout << "*********" << std::endl;
    // 返回常量迭代器的元素
    for (auto it = vec.cbegin();it != vec.cend();it++)
    {
        std::cout << " " << " : " << *it << std::endl;
    }
    std::cout << "*********" << std::endl;

    // 逆序返回常量迭代器的元素
    for (auto it = vec.rbegin();it != vec.rend();it++)
    {
        std::cout << " " << " : " << *it << std::endl;
    }
    std::cout << "*********" << std::endl;

    std::cout << "size: " << vec.size() << " Capacity: " << vec.capacity() << std::endl;

    vec.clear();
    /* vec.resize(n) resize的扩容不会改变容器中原来的值,这里默认对扩容的部分初始化为0
     * n > capacity 时 ,可以对vector进行扩容,此时 size = capacity = n,n 为任意的大于原来capacity的值
     * n < capacity 时,不能对vector进行扩容,此时 size = n,但是 capacity 仍然与原来的capacity 相等
     * vec.reserve(n) 是指将容器的容量改为n,容器中的数据的个数不做改变也就是,不会对vec.size() 做改变
     * n > capacity 时 ,可以仅仅对容器进行扩容,此时size保持不变,capacity = n
     * n < capacity 时 ,不做任何的改变,对size 和capacity没有任何影响
     * vec.assign(n,0) assign的扩容会改变容器中原来的值,第二个参数就是需要改变后的值
     * n > capacity 时 ,可以对vector进行扩容,此时 size = capacity = n,n 为任意的大于原来capacity的值
     * n < capacity 时,不能对vector进行扩容,此时 size = n,但是 capacity 仍然与原来的capacity 相等
     * 总之,对于vector容器只能增大其容量,不能减小其容量
    */
    vec.push_back(12);
    vec.push_back(13);
    
    // vec.resize(13);
    vec.assign(13,0);
    // vec.reserve(13);  // 仅仅改变capacity 的大小
    std::cout << "Resize size: " << vec.size() << " Capacity: " << vec.capacity() << std::endl;

    std::cout << "size: " << vec.size() << " vec = [ ";
    for (int i = 0;i < vec.size();i++)
    {
        std::cout << vec[i] << " ";
    }
    std::cout << " ] " << std::endl; // vec.assign(13,0) 的输出结果: size: 13 vec = [ 0 0 0 0 0 0 0 0 0 0 0 0 0  ]  vec.resize(13) 的输出结果: size: 13 vec = [ 12 13 0 0 0 0 0 0 0 0 0 0 0  ] 

    for (int i = 0;i < 10;i++)
    {
        vec.push_back(i);
    } // 需要对vector进行扩容,一般扩容是2的指数级别的
    
    std::cout << "After clear size: " << vec.size() << " Capacity: " << vec.capacity() << std::endl;

    if (vec.empty())
    {
        std::cout << "Vec is empty!" << std::endl;
    }

    std::vector<int> vecT[3];// vector 定义二维数组

    for (int i = 0;i < 3;i++)
    {
        vecT[i].push_back(i);
        std::cout << "vecT" << i << " size: " << vecT[i].size() << std::endl;
    }

    std::vector<std::vector<int>> vecT1;// vector 定义二维数组
    vecT1.resize(5);//5 行
    for (int i = 0;i < 5;i++)
    {
        vecT1[i].resize(10);//10 列
    }
    for (int i = 0;i < vecT1.size();i++)
    {
        for (int j = 0;j < vecT1[i].size();j++)
        {
            vecT1[i][j] = i*j;
        }
    }

    for (int i = 0;i < vecT1.size();i++)
    {
        for (int j = 0;j < vecT1[i].size();j++)
        {
            std::cout << vecT1[i][j] << " ";
        }
    }
    std::cout << std::endl;
    

    return 0;
}

5.1.2 vector扩容

(1)vec.resize(n)

        vec.resize(n) resize的扩容不会改变容器中原来的值,这里默认对扩容的部分初始化为0

        n > capacity 时 ,可以对vector进行扩容,此时 size = capacity = n,n 为任意的大于原来capacity的值

        n < capacity 时,不能对vector进行扩容,此时 size = n,但是 capacity 仍然与原来的capacity 相等

(2)vec.reserve(n)

        vec.reserve(n) 是指将容器的容量改为n,容器中的数据的个数不做改变也就是,不会对vec.size() 做改变

        n > capacity 时 ,可以仅仅对容器进行扩容,此时size保持不变,capacity = n

        n < capacity 时 ,不做任何的改变,对size 和capacity没有任何影响

(3)vec.assign(n,0)

        vec.assign(n,0) assign的扩容会改变容器中原来的值,第二个参数就是需要改变后的值

        n > capacity 时 ,可以对vector进行扩容,此时 size = capacity = n,n 为任意的大于原来capacity的值

        n < capacity 时,不能对vector进行扩容,此时 size = n,但是 capacity 仍然与原来的capacity 相等

        总之,对于vector容器只能增大其容量,不能减小其容量

     

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/266072.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Java经典框架之Spring

Java经典框架之Spring Java 是第一大编程语言和开发平台。它有助于企业降低成本、缩短开发周期、推动创新以及改善应用服务。如今全球有数百万开发人员运行着超过 51 亿个 Java 虚拟机&#xff0c;Java 仍是企业和开发人员的首选开发平台。 课程内容的介绍 1. Spring简介 2.…

20231220将NanoPC-T4(RK3399)开发板的Android10的SDK按照Rockchip官方挖掘机开发板编译打包刷机之后启动跑飞

20231220将NanoPC-T4(RK3399)开发板的Android10的SDK按照Rockchip官方挖掘机开发板编译打包刷机之后启动跑飞 2023/12/20 17:19 简略步骤&#xff1a;rootrootrootroot-X99-Turbo:~/3TB$ tar --use-compress-programpigz -xvpf rk3399-android-10.git-20210201.tgz rootrootro…

HarmonyOS4.0系统性深入开发02 UIAbility组件详解(上)

UIAbility组件概述 概述 UIAbility组件是一种包含UI界面的应用组件&#xff0c;主要用于和用户交互。 UIAbility组件是系统调度的基本单元&#xff0c;为应用提供绘制界面的窗口&#xff1b;一个UIAbility组件中可以通过多个页面来实现一个功能模块。每一个UIAbility组件实例…

智能优化算法应用:基于爬行动物算法3D无线传感器网络(WSN)覆盖优化 - 附代码

智能优化算法应用&#xff1a;基于爬行动物算法3D无线传感器网络(WSN)覆盖优化 - 附代码 文章目录 智能优化算法应用&#xff1a;基于爬行动物算法3D无线传感器网络(WSN)覆盖优化 - 附代码1.无线传感网络节点模型2.覆盖数学模型及分析3.爬行动物算法4.实验参数设定5.算法结果6.…

Android Canvas画布saveLayer与对应restoreToCount,Kotlin

Android Canvas画布saveLayer与对应restoreToCount&#xff0c;Kotlin private fun mydraw() {val originBmp BitmapFactory.decodeResource(resources, R.mipmap.pic).copy(Bitmap.Config.ARGB_8888, true)val newBmp Bitmap.createBitmap(originBmp.width, originBmp.heigh…

【数据结构一】初始Java集合框架(前置知识)

Java中的数据结构 Java语言在设计之初有一个非常重要的理念便是&#xff1a;write once&#xff0c;run anywhere&#xff01;所以Java中的数据结构是已经被设计者封装好的了&#xff0c;我们只需要实例化出想使用的对象&#xff0c;便可以操作相应的数据结构了&#xff0c;本篇…

数值分析期末复习

第一章 科学计算 误差 解题步骤 先求绝对误差: ∣ x − x ∗ ∣ |x - x^*| ∣x−x∗∣求相对误差限: ∣ x − x ∗ ∣ x ∗ \frac{|x\,\,-\,\,x^*|}{x^*} x∗∣x−x∗∣​求有效数字 ∣ x − x ∗ ∣ 需要小于它自身的半个单位 |x-x^*|\text{需要小于它自身的半个单位} ∣…

qt简单连接摄像头

要使用摄像头&#xff0c;就需要链接多媒体模块以及多媒体工具模块 需要在.pro文件中添加QT multimedia multimediawidgets 是用的库文件 QCamera 类用于打开系统的摄像头设备&#xff0c; QCameraViewfinder 用于显示捕获的视频&#xff0c; QCameraImageCapt…

3D 纹理贴图基础知识

在线工具推荐&#xff1a; 3D数字孪生场景编辑器 - GLTF/GLB材质纹理编辑器 - 3D模型在线转换 - Three.js AI自动纹理开发包 - YOLO 虚幻合成数据生成器 - 三维模型预览图生成器 - 3D模型语义搜索引擎 介绍 纹理贴图是创建模型时离不开的最后一块拼图。同样&#xff0c;…

陈可之|三峡|《河水不犯井水的游戏》

《河水不犯井水的游戏》 尺寸&#xff1a;130x90cm 陈可之2007年绘 油画《河水不犯井水的游戏》是陈可之先生三峡系列作品之一&#xff0c;巧借古代传说&#xff0c;并具象化为螃蟹、乌龟、长江水&#xff0c;描绘了一幅和谐共生的画面。 画面右侧&#xff0c;蜿蜒宽阔的长江水…

内存函数的学习

额外知识点 第一个 假设c为int类型&#xff0c;&#xff08;char&#xff09;c之后&#xff0c;之后如果还用变量c的话&#xff0c;c依然为int类型。&#xff08;&#xff09;强制转换操作符并不会永久改变原本的变量类型。 第二个 \0在打印时不会显示出来 第三个 void …

Latex生成的PDF中加入书签/Navigation/导航

本文参考&#xff1a;【Latex学习】在生成pdf中加入书签/目录/提纲_latex 书签-CSDN博客 &#xff08;这篇文章写的真的太棒了&#xff01;非常推荐&#xff09; 题外话&#xff0c;我的碎碎念&#xff0c;这也是我如何提高搜索能力的办法&#xff1a;想在Latex生成的PDF中加入…

解决用Fiddler抓包,网页显示你的连接不是专用/私密连接

关键&#xff1a;重置fiddler的证书 在Fiddler重置证书 1、Actions --> Reset All Certificates --> 弹窗一路yes 2、关掉Fiddler&#xff0c;重新打开 3、手机删掉证书&#xff0c;重新下载安装。 &#xff08;如果还不行&#xff0c;重新试一遍&#xff0c;先把浏览器…

用python对航空公司客户价值进行聚类分析

1.实验目的 1.会用Python创建KMeans聚类分析模型&#xff1b; 2.使用KMeans模型对航空公司客户价值进行聚类分析&#xff1b; 3.会对聚类结果进行分析 2.实验设备 Jupyter notebook 3.实验原理 4.实验内容 使用sklearn.cluester的KMeans类对航空公司客户数据进行聚类分析&…

Springboot+vue的装饰工程管理系统(有报告),Javaee项目,springboot vue前后端分离项目

演示视频&#xff1a; Springbootvue的装饰工程管理系统&#xff08;有报告&#xff09;&#xff0c;Javaee项目&#xff0c;springboot vue前后端分离项目 项目介绍&#xff1a; 本文设计了一个基于Springbootvue的前后端分离的装饰工程管理系统&#xff0c;采用M&#xff08…

TSINGSEE青犀可视化视频云平台JT/T1078接入能力在智慧物流中的应用

一、引言 随着科技的快速发展和全球贸易的蓬勃发展&#xff0c;智慧物流成为了现代物流业的重要发展方向。智慧物流通过引入先进的信息技术&#xff0c;实现了物流过程的自动化、智能化和信息化&#xff0c;从而提高了物流效率和准确性。在这个过程中&#xff0c;JT/T1078接入…

天文观测与计算机技术:粉丝的视角

天文观测与计算机技术&#xff1a;粉丝的视角 在浩瀚无垠的宇宙中&#xff0c;每一颗星星都像是神秘的密码&#xff0c;等待我们去破译。作为天文观测的粉丝&#xff0c;我深知每一个美丽的星图背后都离不开计算机技术的支持。在这篇文章中&#xff0c;我将分享我对天文观测与…

蓝桥杯第二场小白入门赛(1~5)(对不起,我线段树太菜了)

1.模拟 2.贪心 3.二分 4.数论 5.数论 6.线段树&#xff08;线段树还是练少了...&#xff09; 1. 蓝桥小课堂-平方和 直接模拟&#xff0c;注意数据范围 #include <bits/stdc.h> using namespace std; #define LL long long #define pb push_back #define x first …

华清远见嵌入式学习——ARM——作业2

目录 作业要求&#xff1a; 现象&#xff1a; 代码&#xff1a; 思维导图&#xff1a; 模拟面试题&#xff1a; 作业要求&#xff1a; GPIO实验——3颗LED灯的流水灯实现 现象&#xff1a; 代码&#xff1a; .text .global _start _start: 设置GPIOEF时钟使能 0X50000…

【Linux驱动】字符设备驱动程序框架 | LED驱动

&#x1f431;作者&#xff1a;一只大喵咪1201 &#x1f431;专栏&#xff1a;《RTOS学习》 &#x1f525;格言&#xff1a;你只管努力&#xff0c;剩下的交给时间&#xff01; 目录 &#x1f3c0;Hello驱动程序⚽驱动程序框架⚽编程 &#x1f3c0;LED驱动⚽配置GPIO⚽编程驱动…