PHP优先级队列

Wesley13
• 阅读 605

优先级队列

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

队列是一种特殊的线性表,特殊之处在于它只允许在表的前端(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
点赞
收藏
评论区
推荐文章
待兔 待兔
5个月前
手写Java HashMap源码
HashMap的使用教程HashMap的使用教程HashMap的使用教程HashMap的使用教程HashMap的使用教程22
22 22
3年前
【数据结构之队列】详细图解!在学习队列?看这一篇就够了!
提要钩玄:本文主要介绍队列的结构、基本原理及操作,涉及到两种实现:顺序队列和链队列。1.什么是队列?先举一个日常例子,排队买饭。大家按先来后到的顺序,在窗口前排队买饭,先到先得,买完之后走开,轮到下一位买,新来的人排在队尾,不能插队。可见,上面的“队”的特点是只允许从一端进入,从另一端离开。这样的一个队,放在数据结构中就是“队列”。首先,队列是一个,所以
Wesley13 Wesley13
3年前
java实现队列的详细代码
一、什么是队列结构一种线性结构,具有特殊的运算法则【只能在一端(队头)删除,在另一端(队尾)插入】。分类:1.顺序队列结构2.链式队列结构基本操作:1.入队列2.出队列  二:准备数据staticfinal intQUEUELEN15;classDATA{          String
似梦清欢 似梦清欢
2年前
栈和队列
栈原理栈(stack)又名堆栈,是一种只能在表尾进行插入和删除操作的线性表。能够进行操作的这一端被称为栈顶,相对地,把另一端称为栈底。:::warning栈内元素操作时先进后出,类似于电梯上下成员,最后进去的人最先出
Wesley13 Wesley13
3年前
Java实现顺序栈
一、分析  栈是限定仅在表的一端进行插入或删除操作的线性表,对于栈来说,操作端称为栈顶,另一端则称为栈底,栈的修改是按照后进先出的原则进行的,因此又称为后进先出的线性表。  顺序栈是指利用顺序存储结构实现的栈,即利用一组地址连续的存储单元依次存放自栈底到栈顶的数据元素,同时附设指针top指示栈顶元素在顺序栈中的位置。  一个标准的顺序栈
Stella981 Stella981
3年前
C++ 优先队列priority_queue用法
头文件:include<queue操作:top访问队头empty队列是否为空size返回队列元素个数push插入元素到队尾pop弹出队头swap交换内容定义:1/2Type数据类型3Container容器类型(必须是vect
Stella981 Stella981
3年前
PriorityBlockingQueue 介绍
PriorityBlockingQueue是一个基于优先级堆的无界的并发安全的优先级队列(FIFO),队列的元素按照其自然顺序进行排序,或者根据构造队列时提供的Comparator进行排序,具体取决于所使用的构造方法。实现原理PriorityBlockingQueue通过使用堆这种数据结构实现将队列中的元素按照某种排序规则进行排序,从而改变先进先
Stella981 Stella981
3年前
ConcurrentQueue队列的基本使用方式
 队列(Queue)代表了一个先进先出的对象集合。当您需要对各项进行先进先出的访问时,则使用队列。当您在列表中添加一项,称为入队,当您从列表中移除一项时,称为出队。  ConcurrentQueue<T队列是一个高效的线程安全的队列,是.NetFramework4.0,System.Collections.Concurren
Wesley13 Wesley13
3年前
Java并发 阻塞队列
阻塞队列(BlockingQueue)是一个支持两个附加操作的队列。这两个附加操作支持阻塞地插入和移除方法。支持阻塞插入的方法是指当队列满时会阻塞插入元素的线程,直到队列不满;支持阻塞移除的方法是指当队列为空时获取元素的线程无法继续获取元素直到队列不空。可以发现阻塞队列非常适合消费者和生产者场景下进行使用,生产者生产数据就是向阻塞队列中插入元素,消费者消
菜园前端 菜园前端
1年前
什么是队列
原文链接:什么是队列?队列是一种遵循先进先出原则的有序集合,添加新元素的一端称为队尾,另一端称为队首。实现功能在JavaScript中没有队列,但是可以通过Array实现队列的所有功能enqueue()入队dequeue()出队top()获取队首值size