priority_queue的使用与模拟实现(容器适配器+stack与queue的模拟实现源码)

priority_queue的使用与模拟实现

  • 引言(容器适配器)
  • priority_queue的介绍与使用
    • priority_queue介绍
    • 接口使用
      • 默认成员函数
    • size与empty
    • top
    • push与pop
  • priority_queue的模拟实现
    • 构造函数
    • size与empty
    • top
    • push与pop
    • 向上调整建堆与向下调整建堆
      • 向上调整建堆
      • 向下调整建堆
  • 源码概览
  • 总结

引言(容器适配器)

在STL中,有这样一种“容器”,自身并不完成对数据的操作,而是以其底部的容器来完成所有对数据的工作,类似于对容器的封装。这种“容器”称为container adapter(容器适配器)

stack(栈),queue(队列),priority_queue(优先级队列)都是容器适配器。
栈与队列的实现大家可以参考之前C语言的栈和队列实现,关于stack与queue容器适配器的实现会放在文末:
戳我看栈的模拟实现哦
戳我看队列的模拟实现哦

这篇文章就来介绍一下优先级队列的使用与实现:
(优先级队列的实现需要具备堆数据结构的知识,可以查看相关文章戳我看堆详解哦,戳我看堆排序详解哦)

priority_queue的介绍与使用

priority_queue介绍

在这里插入图片描述

优先级队列与队列类似,我们不能随意访问其中的元素,只能访问队头的首元素。但是在优先级队列中,根据某种优先级(某个标准),任意时刻队列的首元素都会是整个队列中优先级最高的元素(默认情况下队头的元素为最大的元素,可以通过迪三个模板参数来调整);

优先级队列的底层容器必须实现有empty()size()front()push_back()pop_back()。默认的底层容器为vector;

优先级队列的结构类似于堆结构,堆结构可以使堆顶的元素为所有元素中最大/小的元素(通过建大堆或小堆来实现)。所以优先级队列的底层实现其实就是堆结构。在容器适配器中会在需要的时刻建堆,以保证优先级队列队头的元素始终是整个队列中优先级最高的。

优先级队列有三个模板参数
第一个参数 T即优先级队列中存储的数据类型(默认为vector);
第二个参数 Container即优先级队列使用的底层容器类型
第三个参数 Compare是一个仿函数类型,即优先级的比较接口类型(默认为less<T>,即当第一个参数的优先级小于第二个元素时返回真,与之相反的仿函数为greaterless/greater都是库实现的,底层调用T类型的<>)。

接口使用

默认成员函数

构造
在这里插入图片描述
优先级队列的构造函数有两种重载版本,即无参构造使用迭代器区间构造(在构造函数中会使用指定底层容器与比较仿函数的默认构造函数来初始化一个容器对象仿函数对象,所以我们传参构造的时候可以不用去管它们);
对于迭代器区间构造,这是一个函数模板,即可以通过任意一种InputIterator迭代器来构造优先级队列:

int main()
{
	vector<int> v(10);
	for (int i = 0; i < 10; ++i)
	{
		v[i] = i + 1;
	}

	priority_queue<int> pq1; //无参构造
	priority_queue<int> pq2(v.begin(), v.end()); //迭代器区间构造
	cout << pq2.top() << endl;

	priority_queue<int, vector<int>, greater<int>> pq3(v.begin(), v.end()); //指定优先级
	cout << pq3.top() << endl;

	return 0;
}

在这里插入图片描述
pq1使用无参构造;
pq2使用迭代器vbeginend构造,默认的优先级队列头为最大值;
pq3使用v迭代器区间构造的同时,指定了优先级比较的仿函数,队列头为最小值。

优先级队列的包装器没有定义拷贝构造,赋值运算符重载、析构函数,因为编译器会自动生成,自动生成的这些默认成员函数会去调用底层容器的这些默认成员函数

size与empty

在这里插入图片描述
size用于获取优先级队列中元素的个数;
在这里插入图片描述
empty用于返回优先级队列是否为空:为空返回真,否则返回假:

int main()
{
	vector<int> v(10);
	for (int i = 0; i < 10; ++i)
	{
		v[i] = i + 1;
	}

	priority_queue<int> pq(v.begin(), v.end()); //迭代器区间构造
	cout << pq.size() << endl;
	cout << pq.empty() << endl;

	return 0;
}

在这里插入图片描述

top

在这里插入图片描述
top用于获取优先级队列头的元素:

int main()
{
	vector<int> v(10);
	for (int i = 0; i < 10; ++i)
	{
		v[i] = i + 1;
	}

	priority_queue<int> pq(v.begin(), v.end()); //迭代器区间构造
	cout << pq.top() << endl;

	return 0;
}

在这里插入图片描述

push与pop

在这里插入图片描述
push用于在优先级队列中插入一个元素,由于优先级队列会维持队列头的元素始终优先级最大,所以我们不能进行随机插入:

int main()
{
	vector<int> v(10);
	for (int i = 0; i < 10; ++i)
	{
		v[i] = i + 1;
	}

	priority_queue<int> pq(v.begin(), v.end()); //迭代器区间构造
	cout << "before top:" << pq.top() << endl;

	pq.push(12);
	cout << "after top:" << pq.top() << endl;

	return 0;
}

在这里插入图片描述
在这里插入图片描述
pop用于在优先级队列中删除队列头的元素,即删除优先级最高的那个元素:

int main()
{
	vector<int> v(10);
	for (int i = 0; i < 10; ++i)
	{
		v[i] = i + 1;
	}

	priority_queue<int> pq(v.begin(), v.end()); //迭代器区间构造
	cout << "before top:" << pq.top() << endl;

	pq.pop();
	cout << "after top:" << pq.top() << endl;

	return 0;
}

在这里插入图片描述

priority_queue的模拟实现

为避免命名冲突,我们创建一个新的命名空间qqq来放我们的模拟实现;
作为一个类模板,有三个模板参数;
在优先级队列中需要两个成员变量:底层的容器对象与仿函数对象

namespace qqq 
{
    template <class T, class Container = vector<T>, class Compare = less<T> >
    class priority_queue
    {
    
    private:
        Container c;
        Compare comp;
    };
};

需要提醒的是:这里的ContainerCompare是底层容器与仿函数的类型,用这两个类型创建的ccomp才是容器对象与仿函数对象,在传参或者指定模板参数时不要搞错了。

构造函数

除了构造函数之外的其余默认成员函数,系统生成的默认成员函数会调用底层容器的默认成员函数,就足够了。

对于无参构造,我们其实不需要去实现,但是因为要实现迭代器区间构造,所以编译器不会再生成一个默认构造函数,所以我们写一个接口即可:

    priority_queue()
    {}

对于迭代器区间构造,这是一个函数模板:
我们首先通过循环遍历将迭代器区间中的元素拷贝到优先级队列中的成员容器c中;
然后通过循环调用向下调整建堆的成员函数adjust_down实现队列顶的元素优先级最高(对于已有的数组,向下调整建堆的效率要略高于向上调整建堆,这一点在后面详细介绍这两个建堆函数时会介绍):

    template <class InputIterator>
    priority_queue(InputIterator first, InputIterator last) //向下调整建堆
    {
        while (first != last) //遍历拷贝
        {
            c.push_back(*first);
            ++first;
        }
        for (int i = (c.size() - 2) / 2; i >= 0; --i) //向下调整建堆
        {
            adjust_down(i);
        }
    }

size与empty

sizeempty的实现就是调用底层容器的sizeempty接口即可,这也是为什么要求底层容器必须要定义有这些接口:

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

top

top函数在库实现中,调用了底层容器的front接口,返回值为T&
我们在模拟实现时直接返回c的第一个元素即可:

    T& top() const
    {
        return c[0];
    }

push与pop

对于 push 函数:
首先调用底层容器的push_pack在底层容器尾插入一个元素;
然后使用向上调整建堆将新插入的元素调整至正确的位置,保证队列头的元素为优先级最大的元素:
在这里插入图片描述

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

对于 pop 函数:
因为如果直接将底层容器中的元素整体向前移动一个元素来删除第一个元素的话,就会破坏整个堆结构。所以在这里先将底层容器中的首尾元素互换,这样只破坏了一个元素的堆的位置,并且使要删除的元素正好处于底层容器的尾部。
pop_pack删除尾元素后,再将那个被交换到堆顶的元素通过向下调整建堆移动到正确的位置即可:
在这里插入图片描述
在这里插入图片描述

    void pop() //Remove top element
    {
        //交换后pop,再向下调整
        swap(c[0], c[c.size() - 1]);
        c.pop_back();
        adjust_down(0);
    }

向上调整建堆与向下调整建堆

关于堆这个数据结构的详细知识,可以参考我之前的文章:戳我看堆详解哦

这里只提一个要点,即结点数据的下标与结点位置的规律:
父亲结点的下标 = (孩子结点下标-1)/2;
左孩子结点的下标 = 父亲结点的下标2+1;
右孩子结点的下标 = 父亲结点的下标2+2。

这些规律使我们在根据逻辑结构访问堆中的元素时,会很方便。

向上调整建堆

向上调整建堆就是将元素由下到上调整至合适的位置:
类似于二叉树的尾插,然后将尾插的元素向上调整放到合适的位置(若子节点的值小于父亲结点的值,就将父子结点的值交换,直到再次形成一个大堆):

   void adjust_up(int child)
    {
        assert(0 <= child && child < size());
        while (child > 0)
        {
            int parent = (child - 1) / 2;

            if (comp(c[parent], c[child]))
            {
                swap(c[parent], c[child]);
            }

            child = parent;
        }
    }

向下调整建堆

向下调整建堆就是将元素由上到下调整至合适的位置:
向下调整建堆时,即将每一个根结点向下调整,放在其合适的位置。这要求该根的子树都是正确的大堆,所以我们需要从倒数第二层的根结点开始向下调整(比较父亲结点与较大的子结点的值,若父亲结点小于较大的子节点,则交换它们的值。直到再次形成大堆):

    void adjust_down(int parent)
    {
        assert(0 <= parent && parent < size());
        int bigger_child = parent * 2 + 1;
        while (bigger_child < size())
        {
            if (bigger_child + 1 < size() && comp(c[bigger_child], c[bigger_child + 1]))
            {
                ++bigger_child;
            }
            
            if (comp(c[parent], c[bigger_child]))
            {
                swap(c[parent], c[bigger_child]);
            }

            parent = bigger_child;
            bigger_child = parent * 2 + 1;
        }
    }

(这里示意图就不画了,在关于堆的详解中有详细的图示)

关于在对整个数组的数据进行建堆时,为何向下调整建堆的效率高于向上调整建堆的效率,这里做简单的说明(大家可以去计算他们的时间复杂度,这里只从逻辑上分析):
面对一个未成形的堆,最后一行的元素个数是最多的,向上调整建堆从最后一层的最后一个元素向上调整,每个元素最大需要调整高度次。即最多的元素需要调整最多的次数
而向下调整建堆从倒数第二行的最后一个元素开始向下调整,每个元素最大只需要调整一次。即最多的元素需要调整最少的次数。在向下调整建堆中,最多需要调整高度次的元素只有一个,那就是堆顶的元素。
所以向下调整建堆相对于向上调整建堆的效率较高,但他们都属于n*logn级的时间复杂度,效率都很高

源码概览

priority_queue模拟实现源码:

namespace qqq 
{
    template <class T, class Container = vector<T>, class Compare = less<T> >
    class priority_queue
    {
    private:
        void adjust_up(int child)
        {
            assert(0 <= child && child < size());

            while (child > 0)
            {
                int parent = (child - 1) / 2;

                if (comp(c[parent], c[child]))
                {
                    swap(c[parent], c[child]);
                }

                child = parent;
            }
        }
        void adjust_down(int parent)
        {
            assert(0 <= parent && parent < size());

            int bigger_child = parent * 2 + 1;
            while (bigger_child < size())
            {
                if (bigger_child + 1 < size() && comp(c[bigger_child], c[bigger_child + 1]))
                {
                    ++bigger_child;
                }

                if (comp(c[parent], c[bigger_child]))
                {
                    swap(c[parent], c[bigger_child]);
                }

                parent = bigger_child;
                bigger_child = parent * 2 + 1;
            }
        }
    public:
        priority_queue()
        {}

        template <class InputIterator>
        priority_queue(InputIterator first, InputIterator last) //向下调整建堆
        {
            while (first != last)
            {
                c.push_back(*first);
                ++first;
            }
            for (int i = (c.size() - 2) / 2; i >= 0; --i)
            {
                adjust_down(i);
            }
        }
        bool empty() const
        {
            return c.empty();
        }
        size_t size() const
        {
            return c.size();
        }
        T& top() const
        {
            return c[0];
        }
        void push(const T& x)
        {
            c.push_back(x);
            adjust_up(c.size() - 1);
        }
        void pop() //Remove top element
        {
            //交换后pop,再向下调整
            swap(c[0], c[c.size() - 1]);
            c.pop_back();
            adjust_down(0);
        }

		//测试代码
        //void print()
        //{
        //    for (auto e : c)
        //    {
        //        cout << e << " ";
        //    }
        //}
    private:
        Container c;
        Compare comp;
    };

};

stack与queue模拟实现源码:

namespace qqq
{

    //stack
    template<class T, class Con = deque<T>>
    class stack
    {
    public:

        stack()
        {}

        void push(const T& x)
        {
            _c.push_back(x);
        }
        void pop()
        {
            if (_c.empty())
                return;
            _c.pop_back();
        }
        T& top()
        {
            return _c.back();
        }
        const T& top()const
        {
            return _c.back();
        }
        size_t size()const
        {
            return _c.size();
        }
        bool empty()const
        {
            return _c.empty();
        }

    private:
        Con _c;
    };

    //queue
    template<class T, class Con = deque<T>>
    class queue
    {
    public:

        queue()
        {}

        void push(const T& x)
        {
            _c.push_back(x);
        }
        void pop()
        {
            if (_c.empty())
                return;
            _c.pop_front();
        }
        T& back()
        {
            return _c.back();
        }
        const T& back()const
        {
            return _c.back();
        }
        T& front()
        {
            return _c.front();
        }
        const T& front()const
        {
            return _c.front();
        }
        size_t size()const
        {
            return _c.size();
        }
        bool empty()const
        {
            return _c.empty();
        }
    private:
        Con _c;
    };
};

总结

到此,关于priority_queue的使用与实现的所有知识就介绍完了
由于堆的知识在之前C语言部分有详细介绍,所以在这篇文中介绍的不是特别详细,大家有需要可以跳转去相关文章,蟹蟹支持

如果大家认为我对某一部分没有介绍清楚或者某一部分出了问题,欢迎大家在评论区提出

如果本文对你有帮助,希望一键三连哦

希望与大家共同进步哦

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

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

相关文章

ssh: connect to host github.com port 22: Connection refused

ssh: connect to host github.com port 22: Connection refused 问题现象 本文以Windows系统为例进行说明&#xff0c;在个人电脑上使用Git命令来操作GitHub上的项目&#xff0c;本来都很正常&#xff0c;突然某一天开始&#xff0c;会提示如下错误ssh: connect to host gith…

通讯录项目的实现以及动态顺序表(基于顺序表)

首先我们要知道什么是顺序表: 顺序表的底层结构是数组,对数组的封装,实现了常⽤的增删改查等接⼝,顺序表分为静态顺序表(使⽤定⻓数组存储元素)和动态顺序表(按需申请) 静态顺序表缺点: 空间给少了不够⽤,给多了造成空间浪费 拿出来我之前以及写好了的顺序表的代码:…

LeetCode、162. 寻找峰值【中等,最大值、二分】

文章目录 前言LeetCode、162. 寻找峰值【中等&#xff0c;最大值、二分】题目及类型思路及代码思路1&#xff1a;二分思路2&#xff1a;寻找最大值 资料获取 前言 博主介绍&#xff1a;✌目前全网粉丝2W&#xff0c;csdn博客专家、Java领域优质创作者&#xff0c;博客之星、阿…

微信小程序(七)navigator点击效果

注释很详细&#xff0c;直接上代码 上一篇 新增内容&#xff1a; 1.默认效果 2.无效果 3.激活效果 源码&#xff1a; index.wxml //如果 <navigator url"/pages/logs/logs">跳转到log页面&#xff08;默认&#xff09; </navigator><navigator url&q…

ctfshow命令执行(web29-web52)

目录 web29 web30 web31 web32 web33 web34 web35 web36 web37 web38 web39 web40 web41 web42 web43 web44 web45 web46 web47 web48 web49 web50 web51 web52 web29 <?php error_reporting(0); if(isset($_GET[c])){$c $_GET[c];if(!preg_match…

Rust基础语法1

所有权转移&#xff0c;Rust中没有垃圾收集器&#xff0c;使用所有权规则确保内存安全&#xff0c;所有权规则如下&#xff1a; 1、每个值在Rust中都有一个被称为其所有者&#xff08;owner&#xff09;的变量&#xff0c;值在任何时候只能有一个所有者。 2、当所有者离开作用域…

MacBookPro怎么数据恢复? mac电脑数据恢复?

使用电脑的用户都知道&#xff0c;被删除的文件一般都会经过回收站&#xff0c;想要恢复它直接点击“还原”就可以恢复到原始位置。mac电脑同理也是这样&#xff0c;但是“回收站”在mac电脑显示为“废纸篓”。 如果电脑回收站&#xff0c;或者是废纸篓里面的数据被清空了&…

【GitHub项目推荐--微软开源的可视化工具】【转载】

说到数据可视化&#xff0c;大家都很熟悉了&#xff0c;设计师、数据分析师、数据科学家等&#xff0c;都需要用各种方式各种途径做着数据可视化的工作.....当然许多程序员在工作中有时也需要用到一些数据可视化工具&#xff0c;如果工具用得好&#xff0c;就可以把原本枯燥凌乱…

springcloud Client端cloud-consumer-order80

文章目录 简介建立module修改pom修改yml主启动类把公共代码写在一个mudule 里面测试 简介 这个是和之前的8001相互配合端口测试 这里的80的用户测试端口。 代码在&#xff1a;GitHub 上&#xff1a;https://github.com/13thm/study_springcloud/tree/main/days2 建立module …

C++无锁队列的原理与实现

目录 1.无锁队列原理 1.1.队列操作模型 1.2.无锁队列简介 1.3.CAS操作 2.无锁队列方案 2.1.boost方案 2.2.ConcurrentQueue 2.3.Disruptor 3.无锁队列实现 3.1.环形缓冲区 3.2.单生产者单消费者 3.3.多生产者单消费者 3.4.RingBuffer实现 3.5.LockFreeQueue实现 …

django admin后台中进行多个手机号解密消耗时间对比

需求&#xff1a; 1 手机号在数据库中是使用rsa方式加密存储&#xff0c;后台查看中需要转换为明文&#xff0c;因为需要解密多个手机号&#xff0c;所以在后台查看中消耗时间3秒&#xff0c;希望通过多线程&#xff0c;多进程&#xff0c;异步方式来缩短时间 相关注意点&…

如何实现查找附近的人-GEO

背景 打开美团&#xff0c;可以通过自身定位查看附近的商品。打开社交软件&#xff0c;可以查看附近的人交友。打开滴滴&#xff0c;可以查看的附近的共享单车&#xff0c;那这些是如何实现&#xff1f; Redis GEO Redis GEO 主要用于存储地理位置信息&#xff0c;并对存储的…

Ubuntu系统Git的安装配置及使用笔记(更新中)

Ubuntu下Git的下载及配置 (1)、下载git 打开终端命令窗口,输入&#xff1a;sudo apt-get install git 提示&#xff1a;sudo命令是用来以其他身份来执行命令&#xff0c;预设的身份为root,使用sudo时必须先输入密码 (2)、可以使用命令git --version查看git的版本号 (3)、设置…

排序链表(LeetCode 148)

文章目录 1.问题描述2.难度等级3.热门指数4.解题思路参考文献 1.问题描述 给你链表的头结点 head &#xff0c;请将其按 升序 排列并返回 排序后的链表 。 示例 1&#xff1a; 输入&#xff1a;head [4,2,1,3] 输出&#xff1a;[1,2,3,4]示例 2&#xff1a; 输入&#xff…

git中合并分支时出现了代码冲突怎么办

目录 第一章、Git代码冲突介绍1.1&#xff09;什么是Git代码冲突①git merge命令介绍②代码冲突原因 1.2&#xff09;提示代码冲突的两种情况①本地不同分支的文件有差异时&#xff1a;②本地仓库和git远程仓库的文件有差异时&#xff1a; 1.3&#xff09;解决合并时的代码冲突…

【冥想X理工科思维】场景7:背锅挨批后…

冥想音频合集&#xff1a;职场解压冥想音频 压力场景&#xff1a; 明明不是我的错&#xff0c;却不得不替领导背锅&#xff0c;受到大老板和其它团队领导的疯狂指责&#xff0c;如何借助冥想&#xff0c;处理批评后的负面情绪&#xff1f; 点击看大图&#xff1a; 详细说明&…

HBase学习五:运维排障

1、负载均衡 1.1 Rgion迁移 在当前的HBase版本中,Region迁移虽然是一个轻量级操作,但实现逻辑依然比较复杂,≈复杂性主要表现在两个方面:其一,Region迁移过程涉及多种状态的改变;其二,迁移过程中涉及Master、ZooKeeper(ZK)以及RegionServer等多个组件的相互协调。 …

JS执行顺序

众所周知&#xff0c;JavaScript 是单线程语言,只能同时执行做一件事(js只有一个线程&#xff0c;称之为main thread-主线程) 1.Javascript 运行机制 main thread 主线程和 call-stack 调用栈(执行栈)&#xff0c;所有的任务都会被放到调用栈等待主线程执行。 2.Javascript 任…

Addressables(2) ResourceLocation和AssetReference

IResourceLocation var op Addressables.LoadResourceLocationsAsync(key); var result op.WaitForCompletion(); 把加载的Key塞进去&#xff0c;不难看出&#xff0c;IResourceLocation可以用来获得资源的详细信息 很适合用于更新分析&#xff0c;或者一些检查工具 AssetR…

HTML动态房屋装饰特效

下面是代码&#xff1a; <!DOCTYPE html> <html lang"en" ><head><meta charset"UTF-8"><title>HTML5房屋装饰工具DEMO演示</title><link rel"stylesheet" href"css/style.css"></he…