【C++】Stack Queue 仿函数

news2025/4/15 10:17:08

📝前言:
这篇文章我们来讲讲STL中的stack和queue。因为前面我们已经有了string、vector和list的学习基础,所以这篇文章主要关注一些stackqueue的细节问题,以及了解一下deque(缝合怪)和priority_queue ,并且模拟实现priority_queue

🎬个人简介:努力学习ing
📋个人专栏:C++学习笔记
🎀CSDN主页 愚润求学
🌄其他专栏:C语言入门基础,python入门基础,python刷题专栏


文章目录

  • 一,Stack && queue
    • 1. 用vector 适配 Stack
    • 2. 用list模拟实现queue
    • 3. 简单认识deque
  • 二,priority_queue
    • 1. 认识优先级队列
    • 2. 仿函数
    • 3. 模拟实现priority_queue

一,Stack && queue

  1. stackqueue其实是container adaptor(容器适配器)
    在这里插入图片描述
    在STL里面他们是用deque来适配的。也就是通过deque来封装,内部实际上用的是deque的接口。

  2. stack.top()返回的是栈顶元素的引用,queue.front()一样

  3. stack.pop()并不会返回值,而是直接pop掉栈顶元素,queue.pop()一样

除了deque能做适配器以外,其他的容器也都可以,比如vector和list

1. 用vector 适配 Stack

对于Stack而言,要实现的是同一边删除与插入的操作,而vector里面正好有pop_back和push_back 这样的接口。

#include<vector>

namespace tr
{
	template<typename T>
	class stack
	{
	public:
		stack(){}
		void push(const T& x) { _a.push_back(x); }
		void pop() { _a.pop_back(); }
		const T& top() const { return _a.back(); }
		T& top() { return _a.back(); }
		size_t size() { return _a.size(); }
		bool empty() { return _a.empty(); }

	private:
		std::vector<T> _a; // 栈的底层用数组
	};
}

测试代码:

#include<iostream>
#include<vector>
#include"Stack.h" 

using namespace std;

void test_Stack() {
	tr::stack<int> st;
	st.push(1);
	st.push(2);
	st.push(3);
	st.push(4);
	st.push(5);
	while (!st.empty())
	{
		cout << st.top() << endl;
		st.pop();
	}

	cout << st.empty() << st.size() << endl;
}

int main() {

	test_Stack();
	return 0;
}

注意头文件和using namespace std;的位置问题:头文件展开时会向上找标识符,比如“Stack.h”用了一个cout,但是using namespace std;在下面,向上找不到就会报错编译错误:“未定义标识符”


2. 用list模拟实现queue

list要满足的要求是一边插入一边删除,由于vector没有头删,所以这时候选择list是更好的

模拟实现:

#pragma once
#include<list>

namespace tr
{
	template<typename T>
	class queue
	{
	public:
		queue() {}
		void push(const T& x) { _a.push_back(x); }
		const T& front() const { return _a.front(); }
		T& front() { return _a.front(); }
		void pop() { _a.pop_front(); } // 头删
		size_t size() { return _a.size(); }
		bool empty() { return _a.empty(); }

	private:
		std::list<T> _a;
	};


}

测试代码:

#include<iostream>
#include"Queue.h" 

using namespace std;

void test_Queue() {
	tr::queue<int> ls;
	ls.push(1);
	ls.push(2);
	ls.push(3);
	ls.push(4);
	ls.push(5);
	while (!ls.empty())
	{
		cout << ls.front() << endl;
		ls.pop();
	}

	cout << "empty: " << ls.empty() << endl << "size: " << ls.size() << endl;
}

int main() {

	test_Queue();
	return 0;
}

运行结果:
在这里插入图片描述


3. 简单认识deque

deque是双端队列,即两边都可以插入和删除

deque并不是真正连续的空间,而是由一段段连续的小空间拼接而成的,实际deque类似于一个
动态的二维数组,其底层结构如下图所示:
在这里插入图片描述
map中控是一个指针数组,每个指针指向一个数组(每个数组大小一样),这些数组才是存储数据真正的地方。
迭代器由四个部分组成:

  • 给定一个“下标” x 找到容器中对应的数据:先 x / n:找到对应的数组编号,再 x % n 找到在组内的位置
  • 判断是否到达一个数组的尾部:cur == last

deque 和 vector 以及 list 的比较:

  1. 头插尾插效率更高
  2. 下标随机访问比vector差一点
  3. 中间插入数据效率低,因为要移动数据

由因为:

  1. stack和queue没有迭代器,不需要访问
  2. 实现stack时:deque的扩容效率比vector高
  3. 实现queue时:一次性申请一块数组,在queue元素个数增长时,不需要想list一样一个个申请,效率更高,且内存利用率更高

所以,stack和queue用了deque做适配器。


二,priority_queue

1. 认识优先级队列

priority_queue:优先队列,也在头文件< queue > 里面
意思是:在使用top()pop()的时候会取优先级高的,默认是大的元素优先级高。(简单来说就是降序)
底层实现时堆,而堆的底层是数组。
在这里插入图片描述

简单使用一下:

int main() {
	priority_queue<int> pq;
	pq.push(3);
	pq.push(2);
	pq.push(6);
	pq.push(1);
	pq.push(8);
	while (!pq.empty())
	{
		cout << pq.top();
		pq.pop();
	}
	return 0;
}

运行结果:
在这里插入图片描述


2. 仿函数

仿函数是一个类,但是可以像调用函数一样去调用这个类,作为回调函数使用。通过重载()来实现

仿函数使用示例:

class Adder {
public:
    // 重载 () 运算符
    int operator()(int a, int b) const {
        return a + b;
    }
};

int main() {
    Adder adder;
    // 像调用函数一样调用仿函数对象
    int result = adder(3, 4); 
    // 或者用匿名对象:Adder()(3, 4) Adder()——创建匿名对象,(3 ,4)调用重载的()
    std::cout << "Result: " << result << std::endl;
    return 0;
}

3. 模拟实现priority_queue

priority_queue头文件:

#pragma once
#include<iostream>
#include<vector>
using namespace std;

template<class T>
class Less
{
public:
	bool operator()(const T& a, const T& b)
	{
		return a < b;
	}
};

template<class T>
class Greater
{
public:
	bool operator()(const T& a, const T& b)
	{
		return a > b;
	}
};

namespace tr
{
	template<class T, class Container = vector<T>, class Compare = Less<T>>
	// Compare 就是比较方法
	class priority_queue
	{
	public:
		void Adjustup(size_t child)
		{
			Compare com; // 构造仿函数对象
			size_t parent = (child - 1) / 2;
			while (child > 0)
			{
				if (com(_a[child], _a[parent])) // 用仿函数对象调用仿函数
				{
					std::swap(_a[child], _a[parent]);
					child = parent;
					parent = (child - 1) / 2;
				}
				else {
					break;
				}
			}
		}

		void Adjustdown(size_t parent)
		{
			Compare com;
			size_t child = parent * 2 + 1;
			while (child < _a.size())
			{
				if (child + 1 < _a.size() && com(_a[child + 1], _a[child]))
				{
					child++;
				}
				if (com(_a[child], _a[parent])) // 相当于孩子节点小于父亲
				{
					std::swap(_a[child], _a[parent]);
					parent = child;
					child = parent * 2 + 1;
				}
				else {
					break;
				}
			}
		}

		priority_queue(){}
		void push(const T& x)
		{
			_a.push_back(x);
			Adjustup(_a.size() - 1);
		}
		T& top()
		{
			return _a[0];
		}
		const T& top() const
		{
			return _a[0];
		}
		void pop()
		{
			std::swap(_a[0], _a[_a.size() - 1]);
			_a.pop_back();
			Adjustdown(0);
		}
		size_t size() { return _a.size(); }
		bool empty() { return _a.empty(); }
	private:
		Container _a;
	};


};

测试代码:

#include"priority_queue.h"
int main() {
	tr::priority_queue<int, vector<int>, Greater<int>> pq; // 传入的不是less,而是Less<int>,类模板传的是类型,函数模板传才是参数
	pq.push(3);
	pq.push(2);
	pq.push(6);
	pq.push(1);
	pq.push(8);
	while (!pq.empty())
	{
		cout << pq.top();
		pq.pop();
	}
	cout << endl;
	return 0;
}

运行结果(大根堆,降序):
在这里插入图片描述

补充小知识点:编译器对模板是按需实例化,首先编译时:只会检查模板的大框架,不会检查类里面函数的内部。第二,当没有使用到类中的成员函数时,编译器在实例化的时候就不会实例化这些函数。(所以有的时候可能类的成员函数有问题,只是没使用到)


🌈我的分享也就到此结束啦🌈
要是我的分享也能对你的学习起到帮助,那简直是太酷啦!
若有不足,还请大家多多指正,我们一起学习交流!
📢公主,王子:点赞👍→收藏⭐→关注🔍
感谢大家的观看和支持!祝大家都能得偿所愿,天天开心!!!

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

代码随想录_单调栈

代码随想录_单调栈 739.每日温度 739. 每日温度 给定一个整数数组 temperatures &#xff0c;表示每天的温度&#xff0c;返回一个数组 answer &#xff0c;其中 answer[i] 是指对于第 i 天&#xff0c;下一个更高温度出现在几天后。如果气温在这之后都不会升高&#xff0c;…

BoostSearch搜索引擎项目 —— 测试用例设计 + web自动化测试代码

web自动化代码&#xff1a; https://gitee.com/chicken-c/boost-search/tree/master/AutoTest

【Ansible自动化运维】一、初步了解,开启自动化运维之旅

在当今数字化时代&#xff0c;随着企业 IT 基础设施规模的不断扩大&#xff0c;传统的手工运维方式逐渐显得力不从心。自动化运维技术应运而生&#xff0c;其中 Ansible 凭借其简洁易用、功能强大的特点&#xff0c;成为众多运维工程师和开发人员的首选工具。本篇文章将从基础概…

条件概率、概率乘法公式、全概率公式和贝叶斯 (Bayes) 公式

定义 设 P ( A ) > 0 P(A) > 0 P(A)>0&#xff0c;若在随机事件 A A A发生的条件下随机事件 B B B发生的概率记作 P ( B ∣ A ) P(B|A) P(B∣A)&#xff0c;定义 P ( B ∣ A ) P ( A B ) P ( A ) P(B|A) \frac{P(AB)}{P(A)} P(B∣A)P(A)P(AB)​ 则称 P ( B ∣ A ) …

kotlin,Android,jetpack compose,日期时间设置

AI生成&#xff0c;调试出来学习&#xff0c;这些小组件会用了&#xff0c;就可以组合一个大点的程序了。 package com.example.mydatetimeimport android.app.AlertDialog import android.os.Bundle import androidx.activity.ComponentActivity import androidx.activity.co…

ASP.NET图书馆借阅系统(源码+lw+部署文档+讲解),源码可白嫖!

摘要 近些年来&#xff0c;随着科技的飞速发展&#xff0c;互联网的普及逐渐延伸到各行各业中&#xff0c;给人们生活带来了十分的便利&#xff0c;图书馆借阅系统利用计算机网络实现信息化管理&#xff0c;使图书信息、图书借阅、归还的管理发展和服务水平有显著提升。 本文拟…

vi/vim常用快捷键

那么今天我们继续昨天没有介绍完的vi编辑器,来看看常用的一些快捷键,方便我们对文件的编辑. 1.拷贝当前行yy,拷贝当前行向下的5行5yy,并粘贴(输入p) 2.删除当前行dd,删除当前行向下的5行5d 3.在文件中查找某个单词[命令模式/关键字,回车查找,输入n就是查找下一个] ⭐️&…

opencv无法设置禁用RGB转换问题

树莓派连接摄像头,摄像头输出格式为YUYV(YUV422)。 通过执行 v4l2-ctl --list-formats --device/dev/video0 可以看的具体的摄像头的数据格式。 使用opencv获取视频流&#xff0c;通过cap.set(cv2.CAP_PROP_CONVERT_RGB, 0)设置禁用自动转换RGB格式&#xff0c;但是打印输出…

MCP+Blender创建电力塔

MCP&#xff08;Model Context Protocol&#xff09;与Blender的结合是当前AI与3D建模领域的热门技术&#xff0c;它通过协议化的方式让Claude等AI模型直接控制Blender&#xff0c;实现自动化3D建模。 1. 功能与原理 • 核心能力&#xff1a;用户通过自然语言指令&#xff08;…

Selenium自动化:玩转浏览器,搞定动态页面爬取

嘿&#xff0c;各位爬虫爱好者和自动化达人们&#xff01;是不是经常遇到这种情况&#xff1a;信心满满地写好爬虫&#xff0c;requests一把梭&#xff0c;结果抓下来的HTML里&#xff0c;想要的数据空空如也&#xff1f;定睛一看&#xff0c;原来数据是靠JavaScript动态加载出…

QAI AppBuilder 快速上手(8): 图像修复应用实例2

LaMa-Dilated模型旨在通过扩张卷积技术实现高效的图像擦除和修复。该模型采用先进的卷积神经网络架构&#xff0c;能够处理复杂的图像输入&#xff0c;并填补图像中的缺失部分&#xff0c;使修复后的图像更加自然和逼真。LaMa-Dilated不仅在图像编辑领域表现出色&#xff0c;还…

【计网】作业4

一. 单选题&#xff08;共22题&#xff0c;64分&#xff09; 1. (单选题)主机甲采用停止-等待协议向主机乙发送数据&#xff0c;数据传输速率是4kb/s&#xff0c;单向传播时延为30ms&#xff0c;忽略确认帧的发送时延。当信道利用率等于80%时&#xff0c;数据帧的长度为&#…

MPDrive:利用基于标记的提示学习提高自动驾驶的空间理解能力

25年4月来自南方科技大学、百度、英国 KCL和琶洲实验室&#xff08;广东 AI 和数字经济实验室&#xff09;的论文“MPDrive: Improving Spatial Understanding with Marker-Based Prompt Learning for Autonomous Driving”。 自动驾驶视觉问答&#xff08;AD-VQA&#xff09;…

【学习笔记】HTTP和HTTPS的核心区别及工作原理

一、基础概念 HTTP&#xff08;超文本传输协议&#xff09;&#xff1a;明文传输数据&#xff0c;默认端口80&#xff0c;容易被窃听或篡改。 HTTPS&#xff08;HTTP SSL/TLS&#xff09;&#xff1a;通过加密传输数据&#xff0c;默认端口443&#xff0c;保障安全性。 二、…

【STL】list介绍(附与vector的比较)

文章目录 1.关于list2.使用2.1 list的构造2.2 list 迭代器的使用2.3 list 容量操作2.3.1 size()2.3.2 empty()2.3.3 resize() 2.4 list 元素访问2.4.1 front()2.4.2 back() 2.5 list 修改操作2.5.1 push_front()2.5.2 pop_front()2.5.3 push_back()2.5.4 pop_back()2.5.5 inser…

Ansible:roles角色

文章目录 Roles角色Ansible Roles目录编排Roles各目录作用创建 roleplaybook调用角色调用角色方法1&#xff1a;调用角色方法2&#xff1a;调用角色方法3&#xff1a; roles 中 tags 使用实战案例 Roles角色 角色是ansible自1.2版本引入的新特性&#xff0c;用于层次性、结构化…

找不到导入的项目“xxx\QtMsBuild\Qt.props”。请确认 Import 声明“$(QtMsBuild)\Qt.props”中计算结果为

系列文章目录 文章目录 系列文章目录前言一、问题原因 前言 新建的项目visual studio2022 使用Qt vs tools 找不到导入的项目“E:\osgEarth\DigitalSimulationPlatform\DigitalSimulationPlatform\QtMsBuild\Qt.props”。 请确认 Import 声明“$(QtMsBuild)\Qt.props”中计算结…

2025 年福建交安安全员考试:结合本省交通特点备考​

福建地处东南沿海&#xff0c;交通建设具有独特特点&#xff0c;这对交安安全员考试备考意义重大。在桥梁建设方面&#xff0c;由于面临复杂的海洋环境&#xff0c;桥梁的防腐、防台风等安全措施成为重点。考生在学习桥梁施工安全知识时&#xff0c;要特别关注福建本地跨海大桥…

UE5 蓝图里的声音

文章目录 支持的格式设置循环播放在场景中放置音频设置音频的衰减与不衰减在UI动画中播放声音使用蓝图节点播放声音按钮本身就可以播放声音 支持的格式 支持&#xff1a;WAV 不支持&#xff1a;MP3 设置循环播放 双击音频&#xff0c;打开音频设置&#xff0c;勾选Looping …

「合诚」携手企企通共建新材料和健康产业采购数智化新生态

在科技革命与产业变革深度融合的时代背景下&#xff0c;新材料与健康产业正迎来数字化、智能化的快速发展。 技术突破与消费升级的双重驱动&#xff0c;推动着行业不断创新&#xff0c;同时也对企业的供应链管理提出了更高要求。 1、合诚&#xff1a;聚焦新材料与健康产业&am…