C++ 容器适配器仿函数与priority_queue的使用

 更新时间:2024年09月26日 08:30:18   作者:FITMT  
本文主要介绍了C++ 容器适配器仿函数与priority_queue的使用,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

容器适配器

适配器是一种设计模式(设计模式是一套被反复使用的、多数人知晓的、经过分类编目的、代码设计经验的总结),该种模式是将一个类的接口转换成客户希望的另外一个接口。 它们的底层都是其他的容器,例如stack和queue的底层容器默认是deque,而priority_queue的底层容器是vector。它们是对这些容器进行了包装提供了用户想要的接口。

仿函数

定义:仿函数不是函数,而是行为类似函数的类,它重载了opprator()

仿函数的优势:

  • 状态维护:仿函数可以持有状态,每次调用可以根据状态改变行为。
  • 内联调用:由于仿函数是通过对象调用的,编译器可以轻易地将其内联,减少调用开销。
  • 高度定制:可以通过对象的属性来调整其行为。

示例:

#include <iostream>
#include <algorithm>
#include <vector>
 
class Compare {
public:
    bool operator()(int a, int b) {
        return a < b;
    }
};
 
int main() {
    std::vector<int> numbers = {10, 65, 30, 99, 23};
 
    // 使用仿函数进行排序
    std::sort(numbers.begin(), numbers.end(), Compare());
 
    std::cout << "Sorted numbers: ";
    for (int num : numbers) {
        std::cout << num << " ";
    }
    std::cout << std::endl;
 
    return 0;
}

在这个示例中,Compare是一个仿函数,它重载了operator()来进行整数比较。我们使用这个仿函数作为std::sort的比较函数,用来对一个整数向量进行排序。

priority_queue

priority_queue简单介绍

priority_queue是一个容器适配器(如stack,queue)它的底层容器是vector。它的行为类似与heap(堆)默认情况下建立的是大堆。它的底层容器必须能支持随时访问任意位置的元素(这也是为了满足建堆的要求)

模拟实现

template<class T>
	struct less
	{
		bool operator()(const T& left, const T& right)
		{
			return left < right;
		}
	};

	template<class T>
	struct greater
	{
		bool operator()(const T& left, const T& right)
		{
			return left > right;
		}
	};

首先实现两个仿函数可以用来建立大(小)堆

然后实现向下和向上调整算法

void AdjustUP(int child)
		{
			int parent = ((child - 1) >> 1);
			while (child)
			{
				if (Compare()(c[parent], c[child]))
				{
					swap(c[child], c[parent]);
					child = parent;
					parent = ((child - 1) >> 1);
				}
				else
				{
					return;
				}
			}
		}

		// 向下调整
		void AdjustDown(int parent)
		{
			size_t child = parent * 2 + 1;
			while (child < c.size())
			{
				// 找以parent为根的较大的孩子
				if (child + 1 < c.size() && Compare()(c[child], c[child + 1]))
					child += 1;

				// 检测双亲是否满足情况
				if (Compare()(c[parent], c[child]))
				{
					swap(c[child], c[parent]);
					parent = child;
					child = parent * 2 + 1;
				}
				else
					return;
			}
		}

向上(向下)调整算法

向上调整算法:从最后一个节点开始与自己的父亲比较,如果比父亲大(小)就和父亲交换位置,直到调整到根节点为止

向下调整算法:从根节点开始,与左右孩子节点中较大(较小)者比较,若根节点比较大(小)则交换位置,一直向下调整到最后一个节点。

接下来就是比较简单的利用接口进行实现

template<class T, class Container = std::vector<T>, class Compare = less<T>>
	class priority_queue
	{
	public:
		// 创造空的优先级队列
		priority_queue() : c() {}

		template<class Iterator>
		priority_queue(Iterator first, Iterator last)
			: c(first, last)
		{
			// 将c中的元素调整成堆的结构
			int count = c.size();
			int root = ((count - 2) >> 1);
			for (; root >= 0; root--)
				AdjustDown(root);
		}

		void push(const T& data)
		{
			c.push_back(data);
			AdjustUP(c.size() - 1);
		}

		void pop()
		{
			if (empty())
				return;

			swap(c.front(), c.back());
			c.pop_back();
			AdjustDown(0);
		}

		size_t size()const
		{
			return c.size();
		}

		bool empty()const
		{
			return c.empty();
		}

		// 堆顶元素不允许修改,因为:堆顶元素修改可以会破坏堆的特性
		const T& top()const
		{
			return c.front();
		}
        private:
		Container c;
	};

到此这篇关于C++ 容器适配器仿函数与priority_queue的文章就介绍到这了,更多相关C++ 容器适配器与priority_queue内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家! 

相关文章

  • c语言将字符串中的小写字母转换成大写字母

    c语言将字符串中的小写字母转换成大写字母

    本文主要介绍了c语言将字符串中的小写字母转换成大写字母的方法实例。具有很好的参考价值。下面跟着小编一起来看下吧
    2017-04-04
  • C++使用ImGUI框架开发一个简单程序

    C++使用ImGUI框架开发一个简单程序

    ImGui 是一个用于C++的用户界面库,跨平台、无依赖,支持OpenGL、DirectX等多种渲染API,下面就跟随小编一起学习一下如何使用ImGUI框架开发一个简单程序吧
    2023-08-08
  • opencv实现矩形检测

    opencv实现矩形检测

    这篇文章主要为大家详细介绍了opencv实现矩形检测,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2020-07-07
  • C++深入讲解类与对象之OOP面向对象编程与封装

    C++深入讲解类与对象之OOP面向对象编程与封装

    学习过C语言的小伙伴知道:C语言是面向过程的,关注的是过程,分析出求解问题的步骤,通过函数调用逐步解决问题,接下来让我们详细的了解
    2022-05-05
  • QT5实现电子时钟

    QT5实现电子时钟

    这篇文章主要为大家详细介绍了QT5实现电子时钟,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-06-06
  • c++实现MD5算法实现代码

    c++实现MD5算法实现代码

    用c++实现了md5算法。包含 md5.h 和md5.cpp 两个文件。主要参考百度百科 “MD5” 原理,代码中变量命名也是参考其中的公式,程序的使用说明在md5.h 文件的末尾注释中
    2013-11-11
  • 基于对话框程序中让对话框捕获WM_KEYDOWN消息的实现方法

    基于对话框程序中让对话框捕获WM_KEYDOWN消息的实现方法

    下面我们将通过程序给大家演示基于对话框的应用程序对WM_KEYDOWN消息的捕获。需要的朋友可以参考下
    2013-05-05
  • C语言scandir函数获取文件夹内容的实现

    C语言scandir函数获取文件夹内容的实现

    scandir 函数用于列举指定目录下的文件列表,本文主要介绍了C语言scandir函数获取文件夹内容的实现,具有一定的参考价值,感兴趣的可以了解一下
    2024-03-03
  • C语言超详细讲解顺序表的各种操作

    C语言超详细讲解顺序表的各种操作

    大家好,今天给大家带来的是顺序表,我觉得顺序表还是有比较难理解的地方的,于是我就把这一块的内容全部整理到了一起,希望能够给刚刚进行学习数据结构的人带来一些帮助,或者是已经学过这块的朋友们带来更深的理解,我们现在就开始吧
    2022-05-05
  • C++的输入与输出和格式化输出

    C++的输入与输出和格式化输出

    这篇文章主要介绍了详解C++中的输入与输出和格式化输出,是C++入门学习中的基础知识,需要的朋友可以参考,希望能够给你带来帮助
    2021-11-11

最新评论