PHP优先级队列

Wesley13
• 阅读 164

优先级队列

首先,我们要了解一下什么叫队列:

队列是一种特殊的线性表,特殊之处在于它只允许在表的前端(front)进行删除操作,而在表的后端(rear)进行插入操作,和栈一样,队列是一种操作受限制的线性表。进行插入操作的端称为队尾,进行删除操作的端称为队头。

从定义来看,队列是无法更改顺序的线性集合。线性集合一般有几种规则:先进先出(队列)、先进后出(栈)。

优先级队列定义如下:

如果我们给每个元素都分配一个数字来标记其优先级,不妨设较小/较大的数字具有较高的优先级,这样我们就可以在一个集合中访问优先级最高的元素并对其进行查找和删除操作了。这样,我们就引入了优先级队列 这种数据结构。 优先级队列(priority queue) 是0个或多个元素的集合,每个元素都有一个优先权,对优先级队列执行的操作有(1)查找(2)插入一个新元素 (3)删除 一般情况下,查找操作用来搜索优先权最大的元素,删除操作用来删除该元素 。对于优先权相同的元素,可按先进先出次序处理或按任意优先权进行。

可以看到,优先级队列对队列进行了优化,从根本上已经改变了进出顺序(当然,若优先级一样的话,则于队列还是一样的处理方式)。

PHP的实现方式

从php5.3起,内部已经实现了优先级队列的类:SplPriorityQueue,注意,官方有一句话:

The SplPriorityQueue class provides the main functionalities of a prioritized queue, implemented using a max heap.

可以看到,php的优先级队列是用大顶堆实现的算法,其初始化时间复杂度为O(n),重排时间复杂度为O(logn)。

具体可以点链接查看,现在我们来看几个重要的方法:

compare

SplPriorityQueue::compare — Compare priorities in order to place elements correctly in the heap while sifting up
public int compare ( mixed $priority1 , mixed $priority2 )

比较优先级,以便在筛选时将元素正确地放置在堆中。

咦,好像JAVA的样子。

setExtractFlags

SplPriorityQueue::setExtractFlags — Sets the mode of extraction
public void SplPriorityQueue::setExtractFlags ( int $flags )

从字面意义上来看就是设置提取标志,参数有如下:

SplPriorityQueue::EXTR_DATA (0x00000001): Extract the data
SplPriorityQueue::EXTR_PRIORITY (0x00000002): Extract the priority
SplPriorityQueue::EXTR_BOTH (0x00000003): Extract an array containing both

分别为处理数据、优先级、两者都进行,怎么理解呢?来搞个demo吧

<?php

class TestPriority extends SplPriorityQueue {
    //值大的优先级大
    public function compare($priority1, $priority2) {
        if($priority1 == $priority2) return 0;
        return $priority1 > $priority2 ? 1 : -1;
    }
}

$p = new TestPriority();
$p->insert('a', 3);
$p->insert('b', 5);
$p->insert('c', 2);
$p->insert('d', 1);
$p->insert('e', 4);
$p->insert('f', 9);
$p->insert('g', 1);

$p->setExtractFlags(SplPriorityQueue::EXTR_BOTH);

if($p->count() > 0) {
    while($p->valid()) {
        $v = $p->current();
        var_dump($v);
        $p->next();
    }
}

$p->insert('a', 3);
$p->insert('b', 5);
$p->insert('c', 2);
$p->insert('d', 1);
$p->insert('e', 4);
$p->insert('f', 9);
$p->insert('g', 1);
$p->setExtractFlags(SplPriorityQueue::EXTR_DATA);

if($p->count() > 0) {
    while($p->valid()) {
        $v = $p->current();
        var_dump($v);
        $p->next();
    }
}

$p->insert('a', 3);
$p->insert('b', 5);
$p->insert('c', 2);
$p->insert('d', 1);
$p->insert('e', 4);
$p->insert('f', 9);
$p->insert('g', 1);
$p->setExtractFlags(SplPriorityQueue::EXTR_PRIORITY);

if($p->count() > 0) {
    while($p->valid()) {
        $v = $p->current();
        var_dump($v);
        $p->next();
    }
}

运行结果如下:

array(2) {
  ["data"]=>
  string(1) "f"
  ["priority"]=>
  int(9)
}
array(2) {
  ["data"]=>
  string(1) "b"
  ["priority"]=>
  int(5)
}
array(2) {
  ["data"]=>
  string(1) "e"
  ["priority"]=>
  int(4)
}
array(2) {
  ["data"]=>
  string(1) "a"
  ["priority"]=>
  int(3)
}
array(2) {
  ["data"]=>
  string(1) "c"
  ["priority"]=>
  int(2)
}
array(2) {
  ["data"]=>
  string(1) "d"
  ["priority"]=>
  int(1)
}
array(2) {
  ["data"]=>
  string(1) "g"
  ["priority"]=>
  int(1)
}
string(1) "f"
string(1) "b"
string(1) "e"
string(1) "a"
string(1) "c"
string(1) "d"
string(1) "g"
int(9)
int(5)
int(4)
int(3)
int(2)
int(1)
int(1)

很明显能够观察到值得变化。

注意

执行处理之前一定要对长度进行判断,否则会出现如下错误:

PHP Fatal error:  Uncaught RuntimeException: Can't peek at an empty heap in /Users/wenglong11/Desktop/priority.php:30
点赞
收藏
评论区
推荐文章
22 22
1年前
【数据结构之链表】看完这篇文章我终于搞懂链表了
一览:本文从零介绍链式存储结构的线性表——单链表。包括以下内容:什么是链式存储存储结构?单链表的结构辨析头结点、头指针等易混淆概念基本的增删改查操作(不带头结点和带头结点)单链表与顺序表的对比线性表的链式存储结构在一文中我们介绍了一种“用曲线连接”的线性表,“曲线”是一种形象化的语言,实际上并不会存在所谓“曲线”的这种东西。所谓“曲线连
22 22
1年前
【数据结构之队列】详细图解!在学习队列?看这一篇就够了!
提要钩玄:本文主要介绍队列的结构、基本原理及操作,涉及到两种实现:顺序队列和链队列。1.什么是队列?先举一个日常例子,排队买饭。大家按先来后到的顺序,在窗口前排队买饭,先到先得,买完之后走开,轮到下一位买,新来的人排在队尾,不能插队。可见,上面的“队”的特点是只允许从一端进入,从另一端离开。这样的一个队,放在数据结构中就是“队列”。首先,队列是一个,所以
小森森 小森森
5个月前
校园表白墙微信小程序V1.0 SayLove -基于微信云开发-一键快速搭建,开箱即用
后续会继续更新,敬请期待2.0全新版本欢迎添加左边的微信一起探讨!项目地址:(https://www.aliyun.com/activity/daily/bestoffer?userCodesskuuw5n)\2.Bug修复更新日历2.情侣脸功能大家不要使用了,现在阿里云的接口已经要收费了(土豪请随意),\\和注意
Stella981 Stella981
1年前
RokectMQ 顺序性 和分布式事务
1.顺序性是根据参数的id来使其同时投递到统一队列上。//RocketMQ通过MessageQueueSelector中实现的算法来确定消息发送到哪一个队列上//RocketMQ默认提供了两种MessageQueueSelector实现:随机/Hash//当然你可以根据业务实现自己的MessageQueueSelecto
Wesley13 Wesley13
1年前
JAVA线程15
一、阻塞队列1\.概述阻塞队列是Java5线程新特征中的内容,Java定义了阻塞队列的接口java.util.concurrent.BlockingQueue。阻塞队列是一个指定长度的队列,如果队列满了,添加新元素的操作会被阻塞等待,直到有空位为止。同样,当队列为空时候,请求队列元素的操作同样会阻塞等待,直到有可用元
Wesley13 Wesley13
1年前
java实现队列的详细代码
一、什么是队列结构一种线性结构,具有特殊的运算法则【只能在一端(队头)删除,在另一端(队尾)插入】。分类:1.顺序队列结构2.链式队列结构基本操作:1.入队列2.出队列  二:准备数据staticfinal intQUEUELEN15;classDATA{          String
Stella981 Stella981
1年前
Python数据结构与算法——双端队列Dequeue
!(https://oscimg.oschina.net/oscnet/fa1ed5efabdf40f390081750e73ed60e.png)点击上方蓝字关注我们双端队列Dequeue双端队列是一种有序的数据集,与队列相似,但双端队列的两端都可以作为队首
Wesley13 Wesley13
1年前
Java并发 阻塞队列
阻塞队列(BlockingQueue)是一个支持两个附加操作的队列。这两个附加操作支持阻塞地插入和移除方法。支持阻塞插入的方法是指当队列满时会阻塞插入元素的线程,直到队列不满;支持阻塞移除的方法是指当队列为空时获取元素的线程无法继续获取元素直到队列不空。可以发现阻塞队列非常适合消费者和生产者场景下进行使用,生产者生产数据就是向阻塞队列中插入元素,消费者消
Suzhou Suzhou
2个月前
线性表
线性表的顺序存储实现(数组形式)称为顺序表。线性表顺序表示原理解析这里描述的线性表是逻辑结构的,独立于存储结构。线性表的顺序表示简称顺序表。顺序表实现线性表的方式是使用数组。线性表第一个元素的数组下标是0。另外一种实现顺序表的方法:使用数组方式比动态分配更
Suzhou Suzhou
3个月前
栈和队列
栈原理栈(stack)又名堆栈,是一种只能在表尾进行插入和删除操作的线性表。能够进行操作的这一端被称为栈顶,相对地,把另一端称为栈底。:::warning栈内元素操作时先进后出,类似于电梯上下成员,最后进去的人最先出