【C++提高编程-05】----C++之Deque容器实战

34889d2e7fec4a02bb0ae3a22b011a2a.png

🎩 欢迎来到技术探索的奇幻世界👨‍💻

📜 个人主页:@一伦明悦-CSDN博客

✍🏻 作者简介: C++软件开发、Python机器学习爱好者

🗣️ 互动与支持💬评论      👍🏻点赞      📂收藏     👀关注+

如果文章有所帮助,欢迎留下您宝贵的评论,

点赞加收藏支持我,点击关注,一起进步!

前言

      Deque(双端队列)是C++ STL中的一种序列容器,具有以下特点和功能:

  1. 双端插入和删除

    • 可以在队列的两端(前端和后端)进行高效的插入和删除操作,时间复杂度为常量时间。
  2. 动态扩展

    • 与向量(std::vector)类似,Deque可以动态增长以容纳更多元素,并支持自动分配更多内存空间。
  3. 随机访问迭代器

    • 提供了随机访问迭代器,允许通过索引快速访问Deque中的任何元素。

正文

01-Deque容器简要解释

      C++中的std::deque(双端队列)是一种序列容器,允许在两端进行高效地插入和删除操作。以下是std::deque容器的详细介绍:

  1. 双端队列概述

    • 双端队列是一种序列容器,类似于向量(std::vector),但允许在队列的两端进行高效地插入和删除操作。
    • 与向量不同,双端队列的插入和删除操作在队列的前端和后端都是常量时间复杂度,而向量只能在末尾进行常量时间的插入和删除。
  2. 容器特点

    • 随机访问迭代器std::deque 提供了随机访问迭代器,可以通过索引快速访问容器中的元素。
    • 动态扩展:与向量一样,双端队列的大小可以动态增长,当容器满时会自动分配更多的内存空间。
    • 双端操作:可以在队列的前端和后端进行插入和删除操作,使得双端队列在某些情况下比向量更加高效。
  3. 成员函数

    • push_front(val): 将元素 val 插入到双端队列的前端。
    • push_back(val): 将元素 val 插入到双端队列的后端。
    • pop_front(): 删除双端队列的第一个元素。
    • pop_back(): 删除双端队列的最后一个元素。
    • front(): 返回双端队列的第一个元素的引用。
    • back(): 返回双端队列的最后一个元素的引用。
    • size(): 返回双端队列中元素的数量。
    • empty(): 检查双端队列是否为空。
    • clear(): 清空双端队列中的所有元素。
  4. 使用场景

    • 当需要在队列的两端频繁地插入和删除元素时,双端队列是一个比较合适的选择。
    • 双端队列通常用于需要快速插入和删除元素的情况,例如实现双端队列数据结构、广度优先搜索等算法中。

        总之,std::deque提供了一种灵活高效的双端队列实现,适用于需要频繁在队列两端进行插入和删除操作的场景。

#include <iostream>
#include <deque>

int main() {
    // 创建一个空的双端队列 deque
    std::deque<int> deque;

    // 在队列的前端和后端插入元素
    deque.push_back(1);    // deque: [1]
    deque.push_front(2);   // deque: [2, 1]
    deque.push_back(3);    // deque: [2, 1, 3]

    // 访问队列的第一个和最后一个元素
    std::cout << "Front element: " << deque.front() << std::endl;  // 输出: 2
    std::cout << "Back element: " << deque.back() << std::endl;    // 输出: 3

    // 弹出队列的第一个和最后一个元素
    deque.pop_front();     // deque: [1, 3]
    deque.pop_back();      // deque: [1]

    // 检查队列是否为空和获取队列的大小
    if (!deque.empty()) {
        std::cout << "Deque is not empty." << std::endl;
        std::cout << "Size of deque: " << deque.size() << std::endl;  // 输出: 1
    }

    // 清空队列
    deque.clear();         // deque: []

    // 再次检查队列是否为空
    if (deque.empty()) {
        std::cout << "Deque is empty now." << std::endl;
    }

    return 0;
}

02-Deque容器介绍

      C++中的std::deque(双端队列)是一种序列容器, 当需要在队列两端频繁地插入和删除元素时,Deque是比较合适的选择。可用于实现双端队列数据结构、广度优先搜索等算法中需要双端操作的情况。总结而言,std::deque提供了一种灵活、高效的双端队列实现,适用于需要在队列两端进行频繁插入和删除操作的场景。

下面给出具体代码分析应用过程:

这段代码演示了如何使用std::deque容器(双端队列)进行元素的插入和遍历操作。下面是对这段代码的详细解释:

  1. 包含头文件及命名空间声明

    • #include <iostream>:包含输入输出流的头文件。
    • using namespace std;:使用命名空间 std
  2. 函数定义

    • printDeque函数:接受一个std::deque<int>的常量引用作为参数,用于打印该双端队列中的元素。函数内部使用const_iterator类型的迭代器来循环遍历双端队列,使用const修饰确保在函数内不会修改容器中的元素。
    • test01函数:在函数内部创建一个空的双端队列 d1,然后使用push_back(尾插法)向队尾插入 0 到 9 十个整数。同时,也提供了push_front(头插法)的方式注释在函数内,向队首插入相同的数据。
  3. 遍历并打印双端队列元素

    • test01函数中,调用printDeque函数打印双端队列 d1 中的所有元素。
  4. main函数

    • main函数中调用test01函数来执行整个测试过程。
    • system("pause");:在Windows系统下暂停控制台窗口,等待用户输入任意键后关闭窗口。
    • return 0;:返回程序执行成功的标志。

总的来说,这段代码展示了如何创建std::deque容器并通过尾插法向其中插入数据,然后通过printDeque函数遍历并打印双端队列中的元素。最后在main函数中进行调用,展示了std::deque的基本使用方法。

#include <iostream>
using namespace std;
#include <deque>


void printDeque(const deque<int>&d)   // 加const防止写操作,那么迭代器那里必须使用const_iterator  const迭代器
{
	for (deque<int>::const_iterator it = d.begin(); it != d.end();it++)
	{
		// *it = 100;   容器中数据不可修改
		cout << *it <<" ";
	}
	cout << endl;
}

void test01()
{

	deque<int>d1;

	for (int i = 0; i < 10;i++)
	{
		d1.push_back(i);   // 尾插法 0,1,2,3,4,5,6,7,8,9
//		d1.push_front(i);   // 头插法,9,8,7,6,5,4,3,2,1,0

	}
	printDeque(d1);


}


int main() {

	test01();

	system("pause");
	return 0;
}

运行结果如下图所示:

 

03-Deque容器排序

       代码演示如下,这段代码演示了如何使用 std::deque 容器进行排序操作,包括手动实现的冒泡排序和使用标准库提供的 sort 函数进行排序。以下是对代码的详细解释:

  1. 包含头文件及命名空间声明

    • #include <iostream>:包含输入输出流的头文件。
    • using namespace std;:使用命名空间 std
    • #include <deque>:包含双端队列 std::deque 的头文件。
    • #include <algorithm>:包含标准库算法的头文件,这里用于使用 sort 函数。
  2. 函数定义

    • printDeque 函数:接受一个 std::deque<int> 的常量引用作为参数,用于打印该双端队列中的元素。函数内部使用 const_iterator 类型的迭代器来循环遍历双端队列,保证不修改容器中的元素。
    • BubbleSort 函数:实现了冒泡排序算法,接受一个 std::deque<int> 的非常量引用作为参数,在函数内部通过交换元素来实现排序。
  3. 测试函数 test01

    • 在函数内部创建了一个空的 std::deque<int>,并依次使用 push_back 和 push_front 方法插入几个整数元素。
    • 调用 printDeque 函数,打印未排序前的双端队列元素。
    • 调用 BubbleSort 函数,对双端队列进行冒泡排序。
    • 再次调用 printDeque 函数,打印冒泡排序后的结果。
    • 使用 std::sort 函数对双端队列进行排序(默认升序),并再次调用 printDeque 打印排序后的结果。
  4. 主函数 main

    • 在 main 函数中调用 test01 函数,执行测试排序功能。
    • system("pause");:在Windows系统下暂停控制台窗口,等待用户输入任意键后关闭窗口。
    • return 0;:返回程序执行成功的标志。

总结来说,这段代码展示了如何使用 std::deque 容器进行排序操作,包括手动实现的冒泡排序和使用标准库提供的 sort 函数进行排序,同时也展示了如何遍历打印双端队列中的元素。

#include <iostream>
using namespace std;
#include <deque>
#include <algorithm>    // 使用所有算法都要包含这个头文件

// deque排序操作

void printDeque(const deque<int>&d1)
{
	for (deque<int>::const_iterator it = d1.begin(); it != d1.end();it++)
	{
		cout << *it << " ";
	}
	cout << endl;
}

void BubbleSort(deque<int>&d1)    // 这如果加了const,就无法对数据进行交换数字操作了
{
	for (int i = 0; i < d1.size()-1;i++)
	{
		for (int j = 0; j < d1.size() - i - 1;j++)
		{
			if (d1[j]>d1[j+1])
			{
				int temp = d1[j];
				d1[j] = d1[j + 1];
				d1[j + 1] = temp;
			}
		}
	}
}

void test01()
{
	deque<int>d1;
	d1.push_back(10);
	d1.push_back(20);
	d1.push_back(30);
	d1.push_front(200);
	d1.push_front(300);

	cout << "排序前" << endl;
	printDeque(d1);

	// 1、使用冒泡排序交换
	BubbleSort(d1);
	cout << "冒泡排序交换后" << endl;
	printDeque(d1);

	// 2、使用sort排序
	sort(d1.begin(), d1.end());   // 默认从小到达排序
	cout << "sort排序交换后" << endl;
	printDeque(d1);

}


int main() {
	test01();

	system("pause");
	return 0;
}

示例运行结果如下图所示:

  

04-Deque容器案例-评委打分

       代码演示如下,这段代码是一个简单的程序,用于创建五个学生对象并打印它们的姓名和分数。下面是对这段代码的详细解释和分析:

  1. 包含头文件及命名空间声明

    • #include <iostream>:包含输入输出流的头文件。
    • using namespace std;:使用命名空间 std
    • #include <string>:包含字符串类的头文件。
    • #include <vector>:包含向量(动态数组)类的头文件。
  2. 类定义

    • class Person:定义了一个名为 Person 的类,用于代表一个学生。
      • Person(string name, int score):构造函数,用于初始化学生的姓名和分数。
      • string m_Name;:学生的姓名。
      • int m_Score;:学生的分数。
  3. 函数定义

    • void creatPerson(vector<Person>&v):函数用于创建五个学生对象并添加到向量 v 中。
      • 使用 for 循环创建五个学生对象。
      • 在循环内部,通过拼接字符串 nameSeed 中的字符来生成学生的姓名。
      • 初始化每个学生的分数为 0
      • 创建 Person 对象并将其添加到向量 v 中。
  4. 主函数 main

    • 在 main 函数中,首先创建一个向量 v 用于存放学生对象。
    • 调用 creatPerson 函数,将创建的学生对象添加到向量 v 中。
    • 使用一个 for 循环遍历向量 v 中的每个学生对象,并打印它们的姓名和分数。
    • system("pause");:在Windows系统下暂停控制台窗口,等待用户输入任意键后关闭窗口。
    • return 0;:返回程序执行成功的标志。

这段代码的作用是创建五个学生对象,每个学生有一个姓名和一个分数,然后将这些学生对象添加到一个向量中,并打印它们的信息。程序没有实现去除最高分和最低分并计算平均分的功能,这一部分需要进一步实现。

#include <iostream>
using namespace std;
#include <string>
#include <vector>
// 要求 有5名选手:选手ABCDE,10个评委分别对每一名选手打分,去除最高分,去除评委中最低分,取平均分。

class Person
{
public:

	Person(string name, int score)
	{
		this->m_Name = name;
		this->m_Score = score;
	}
	
	string m_Name; //姓名
	int m_Score; //平均分
};

void creatPerson(vector<Person>&v)
{
	string nameSeed = "ABCDE";
	for (int i = 0; i < 5;i++)
	{
		
		string name = "学生";
		name += nameSeed[i];
		
		int score = 0;

		Person p (name, score);

		v.push_back(p);
		
	}

}


int main() {

	// 创建容器存放学生
	vector<Person>v;
	creatPerson(v);

	for (vector<Person>::iterator it = v.begin(); it != v.end();it++)
	{
		cout << "名字:" << (*it).m_Name << "分数:" << (*it).m_Score << endl;
	}



	system("pause");
	return 0;
}

运行结果如下图所示:

    

总结

      在C++中,deque(双端队列)是标准模板库中的一种序列容器,具有以下特点和总结:

  1. 双端插入和删除

    • deque 允许在队列的两端(前端和后端)进行高效的插入和删除操作,时间复杂度为常量时间。
  2. 动态扩展

    • 类似于向量(std::vector),deque 可以动态增长以容纳更多元素,并支持自动分配更多内存空间。
  3. 随机访问迭代器

    • deque 提供了随机访问迭代器,允许通过索引快速访问 deque 中的任何元素。
  4. 成员函数

    • push_front(val): 将元素 val 插入到 deque 的前端。
    • push_back(val): 将元素 val 插入到 deque 的后端。
    • pop_front(): 删除 deque 的第一个元素。
    • pop_back(): 删除 deque 的最后一个元素。
    • front(): 返回 deque 的第一个元素的引用。
    • back(): 返回 deque 的最后一个元素的引用。
    • size(): 返回 deque 中元素的数量。
    • empty(): 检查 deque 是否为空。
    • clear(): 清空 deque 中的所有元素。
  5. 使用场景

    • 当需要在队列两端频繁地插入和删除元素时,deque 是比较合适的选择。
    • 可用于实现双端队列数据结构、广度优先搜索等算法中需要双端操作的情况。

       总体而言,deque 提供了一种灵活、高效的双端队列实现,适用于需要在队列两端进行频繁插入和删除操作的场景。

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

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

相关文章

ubuntu vnc

如何在Ubuntu 18.04安装VNC | myfreax sudo apt install xfce4 xfce4-goodies xorg dbus-x11 x11-xserver-utils sudo apt install tigervnc-standalone-server tigervnc-common vncserver sudo apt install xfce4 xfce4-goodies xorg dbus-x11 x11-xserver-utils sudo apt ins…

用Python向Word文档添加页眉和页脚

用Python向Word文档添加页眉和页脚 添加页眉和页脚效果代码 添加页眉和页脚 在本文中&#xff0c;我们将用python向文档中添加页眉和页脚。 效果 添加前的文档&#xff1a; 添加页眉和页脚后&#xff1a; 代码 from docx import Documentdef add_header_footer(doc_path…

【数据结构】——常见排序

文章目录 一、 冒泡排序二、 选择排序三、插入排序四、 快速排序1. hoare版本2. 优化版本3. 前后指针法4. 非递归版本 五、 堆排序六、 希尔排序七、 归并排序1. 递归版本2. 非递归版本 八、 计数排序 在开始之前先准备一个交换数据的函数&#xff0c;排序会经常用到 //交换位置…

牛客热题:兑换零钱(一)

&#x1f4df;作者主页&#xff1a;慢热的陕西人 &#x1f334;专栏链接&#xff1a;力扣刷题日记 &#x1f4e3;欢迎各位大佬&#x1f44d;点赞&#x1f525;关注&#x1f693;收藏&#xff0c;&#x1f349;留言 文章目录 牛客热题&#xff1a;兑换零钱(一)题目链接方法一&am…

解读ROS功能包模块的步骤

系列文章目录 提示:这里可以添加系列文章的所有文章的目录,目录需要自己手动添加 TODO:写完再整理 文章目录 系列文章目录前言解读ROS功能包模块的步骤前言 认知有限,望大家多多包涵,有什么问题也希望能够与大家多交流,共同成长! 推荐开发经验及方法博客专栏: [https:/…

抢占人工智能行业红利,前阿里巴巴产品专家带你15天入门AI产品经理

前言 当互联网行业巨头纷纷布局人工智能&#xff0c;国家将人工智能上升为国家战略&#xff0c;藤校核心课程涉足人工智能…人工智能领域蕴含着巨大潜力&#xff0c;早已成为业内共识。 面对极大的行业空缺&#xff0c;不少人都希望能抢占行业红利期&#xff0c;进入AI领域。…

软件工程实务:软件产品

目录 1、软件产品的基本概念 2、软件工程是什么&#xff1f; 为什么产生软件工程? 软件工程是做什么的? 3、定制软件和软件产品的工程比较 4 、软件产品的运行模式 5、软件产品开发时需要考虑的两个基本技术因素 6、产品愿景 7、软件产品管理 8、产品原型设计 9、小结…

如何区分人工智能生成的图像与真实照片(上)

随着最先进扩散模型&#xff08;如Midjourney、Stable Diffusion和Firefly&#xff09;生成的图像具有高度的逼真度&#xff0c;未经训练的我们很难区分真实照片和AI生成的图像。为了解决这个问题&#xff0c;这份指南&#xff0c;帮助读者培养更批判的眼光&#xff0c;识别AI生…

056、PyCharm 快速代码重构的方法

在实际的编程过程中&#xff0c;如果有一段代码需要在多个地方重复使用&#xff0c;我们应该将这段代码封装成一个函数。这样可以提高代码的可重用性和可维护性。 在PyCharm编辑器里&#xff0c;可以使用以下操作对代码块进行快速的重构。 &#xff08;1&#xff09;、选中一…

【数据分析】推断统计学及Python实现

各位大佬好 &#xff0c;这里是阿川的博客&#xff0c;祝您变得更强 个人主页&#xff1a;在线OJ的阿川 大佬的支持和鼓励&#xff0c;将是我成长路上最大的动力 阿川水平有限&#xff0c;如有错误&#xff0c;欢迎大佬指正 Python 初阶 Python–语言基础与由来介绍 Python–…

CCAA质量管理【学习笔记】​​ 备考知识点笔记(五)质量设计方法与工具

第五节 质量设计方法与工具 1 任 务 分 解 法 1.1 概念 任务分解法&#xff0c;又称工作分解结构 (Work Breakdown Structure, 简 称 WBS) 。WBS 指以可交付成果为 导向&#xff0c;对项目团队为实现项目目标并完成规定的可交付成果而执行的工作所进行的层次分解。W…

mysql 8 创建用户,并对用户授权

创建用户&#xff1a; 对MySQL创建新用户。命令如下&#xff1a; create user devuser% identified by 123456; 授予权限 grant all privileges on joolun_ry.* to devuser% with grant option; 参数说明&#xff1a; joolun_ry&#xff1a;表明对那个库进行授权&#xf…

C语言概述与历史

引言 C语言是一门历史悠久且影响深远的编程语言。它不仅为后继的许多编程语言奠定了基础&#xff0c;同时因其高效性和灵活性在系统编程和嵌入式开发领域得到了广泛应用。本篇文章将全面介绍C语言的起源与发展、设计目标与理念&#xff0c;以及C语言的标准演化历程&#xff0c;…

解决MyBatis获取刚插入数据的ID值

解决MyBatis获取刚插入数据的ID值 Mybatis获取刚插入数据的ID值有很多解决方法&#xff0c;目前采用以下方式进行获取。 添加完数据后直接返回刚添加数据的id // UserDao.java public static void addUser() throws Exception{InputStream resourceAsStream Resources.getR…

学习资料分析

学习资料分析 速算运算 √截位直除分数比较等比修正其他速算方法基期与现期基本概念求基期求现期增长率与增长量增长相关统计术语求一般增长率比较一般增长率增长量比重比重相关公式求比重平均数倍数间隔增长乘积增长率年增长率混合增长率资料分析:主要测查报考者对文字、数字…

N3 中文文本分类

&#x1f368; 本文为&#x1f517;365天深度学习训练营 中的学习记录博客&#x1f356; 原作者&#xff1a;K同学啊# 前言 前言 前面学习了相关自然语言编码&#xff0c;这周进行相关实战 导入依赖库和设置设备 import torch import torch.nn as nn import torchvision fro…

湘潭大学信息与网络安全复习笔记2(总览)

前面的实验和作业反正已经结束了&#xff0c;现在就是集中火力把剩下的内容复习一遍&#xff0c;这一篇博客的内容主要是参考教学大纲和教学日历 文章目录 教学日历教学大纲 教学日历 总共 12 次课&#xff0c;第一次课是概述&#xff0c;第二次和第三次课是密码学基础&#x…

Android入门第68天-自动更新/升级怎么做(生产级实例)

开篇 今天我们进入第68讲。 在第60天左右其实很多同学们已经进入了APP应用开发了,因为60天内容足以让大家踏上正实的Android开发生涯。 随着开发的深入,我们发觉日常工作中无非就是一些组件的嵌套、合理应用。当代码迭代、功能迭代越来越频繁后我们面临着另一个问题,即:…

Vue3 生命周期函数及其与Vue2的对比总结

Vue3 继续保留了 Vue2 的生命周期钩子&#xff0c;但在 Composition API&#xff08;setup 函数&#xff09;中&#xff0c;它们被改为了一组导入函数。以下是它们的对比&#xff1a; Vue2 生命周期钩子和 Vue3 对应的生命周期函数&#xff1a; 在 Vue3 中&#xff0c;所有的…

git 快速将当前目录添加仓储

一、进入目录 git init git add . git commit -m "init" git remote add origin http://192.168.31.104/root/AutoBuildDemo.git 二、登录gitlab&#xff0c;创建项目AutoBuildDemo 最后执行&#xff1a; git push -u origin master