优先级队列实现php,PHP优先级队列 - o0无忧亦无怖的个人空间 - OSCHINA - 中文开源技术交流社区...

优先级队列

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

队列是一种特殊的线性表,特殊之处在于它只允许在表的前端(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吧

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

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
提供的源码资源涵盖了安卓应用、小程序、Python应用和Java应用等多个领域,每个领域都包含了丰富的实例和项目。这些源码都是基于各自平台的最新技术和标准编写,确保了在对应环境下能够无缝运行。同时,源码中配备了详细的注释和文档,帮助用户快速理解代码结构和实现逻辑。 适用人群: 这些源码资源特别适合大学生群体。无论你是计算机相关专业的学生,还是对其他领域编程感兴趣的学生,这些资源都能为你提供宝贵的学习和实践机会。通过学习和运行这些源码,你可以掌握各平台开发的基础知识,提升编程能力和项目实战经验。 使用场景及目标: 在学习阶段,你可以利用这些源码资源进行课程实践、课外项目或毕业设计。通过分析和运行源码,你将深入了解各平台开发的技术细节和最佳实践,逐步培养起自己的项目开发和问题解决能力。此外,在求职或创业过程中,具备跨平台开发能力的大学生将更具竞争力。 其他说明: 为了确保源码资源的可运行性和易用性,特别注意了以下几点:首先,每份源码都提供了详细的运行环境和依赖说明,确保用户能够轻松搭建起开发环境;其次,源码中的注释和文档都非常完善,方便用户快速上手和理解代码;最后,我会定期更新这些源码资源,以适应各平台技术的最新发展和市场需求。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值