stl源码剖析 详细学习笔记heap
//
// heap.cpp
// 笔记
//
// Created by fam on 15/3/15.
//
//
//---------------------------15/03/15----------------------------
//heap
{
/*
heap概述:
heap并不是stl的容器,只是priority queue(优先队列)的助手
它允许用户以任意顺序插入容器,但是取出是,总是取出优先级最高的元素
heap用的是很常见的堆结构,用数组来表示,采用堆排序的方法排序,时刻保持堆的特性
*/
//push_heap
template <class RandomAccessIterator>
inline void push_heap(RandomAccessIterator first,
RandomAccessIterator last)
{
__push_heap_aux(first,last, distance_type(first),
value_type(first));
}
template <class RandomAccessIterator,
class Distance, class T>
inline void __push_heap_aux(RandomAccessIterator first,
RandomAccessIterator last, Distance*,T*)
{
__push_heap(first,Distance((last - first) -
), Distance(),
T(*(last -
)));
}
template <class RandomAccessIterator,
class Distance, class T>
void __push_heap(RandomAccessIterator first, Distance holeIndex,
Distance topIndex, T value)
{
//取最后一个元素的父节点,先减1的原因是第一个节点时0,
//而不是1开始算的
Distance parent = (holeIndex -
) / ;
//一直循环到当前节点等于根节点
或者 父节点的值大于等于添加的值
while (holeIndex > topIndex && *(first + parent) < value)
{
//
把父节点的值赋值给当前节点
*(first + holeIndex) = *(first + parent);
//当前节点变成父节点
holeIndex = parent;
//父节点变成父节点的父节点
parent = (holeIndex -
) / ;
}
//把值赋值给恰当的位置
*(first + holeIndex) = value;
}
//push功能总结:push就是从最后一个元素开始一直和父节点比较,如果要插入的值(最后一个元素的值)
//比较大就不断上移,直到达到堆的根
//pop_heap
template <class RandomAccessIterator>
inline void pop_heap(RandomAccessIterator first,
RandomAccessIterator last)
{
__pop_heap_aux(first,last,value_type(first));
}
template <class RandomAccessIterator,
class T>
inline void __pop_heap_aux(RandomAccessIterator first,
RandomAccessIterator last, T*)
{
__pop_heap(first,last -
,last - , T(*(last -
))
distance_type(first));
}
template <class RandomAccessIterator,
class T, class Distance>
inline void __pop_heap(RandomAccessIterator first,
RandomAccessIterator last,
RandomAccessIterator result,
T value, Distance*)
{
*result = *first;
__adjust_heap(first, Distance(), Distance(last - first), value);
}
template <class RandomAccessIterator,
class Distance, class T>
void __adjust_heap(RandomAccessIterator first, Distance holeIndex,
Distance len, T value)
{
//topIndex == 0 -->> topelem == first + 0
Distance topIndex = holeIndex;
//由于从0开始计算
所以右儿子为 2n + 2;
//secondChild
其实也是Index,不是真的值
Distance secondChild =
* holeIndex + ;
//只要右儿子存在就一直循环
while (secondChild < len)
{
//如果有右儿子小于左儿子,secondChild就设置为左儿子
//目的是要让当前节点最后成为2个节点中最大的
if(*(first + secondChild) < *(first + (secondChild -
)))
secondChild--;//为什么不是 --secondChild,这样效率更高
//把当前节点的值设置为secondChild的值
*(first + holeIndex) = *(first +secondChild);
//当前节点切换成secondChild节点
holeIndex = secondChild;
//设置secondChild为右儿子
secondChild =
* (secondChild + );
}
//存在左儿子(如果有右儿子就会一直循环,最后只有两种情况,一是当前节点到达叶子节点
//一种情况是没有右儿子,但是有左儿子)
if(secondChild == len)
{
//把左儿子的值赋给当前节点
*(first + holeIndex) = *(first + (secondChild -
));
//把当前节点切换成左儿子
holeIndex = secondChild -;
}
//把当初标记的最后的一个节点插入堆中( T(*(last - 1) )
__push_heap(first, holeIndex, topIndex, value);
/*
总结:把根节点输出到result中
所以就要不断把儿子往上移来填充
填充到最后就会空缺一个节点(下面的往上移动总会有一个空缺的)
所以最后要把最后的一个元素插入到堆前面去,也就是最后空缺的是原先
最后面的位置,那就没影响了
如果空缺的已经是最后的元素了,那就是对已经排序好的堆做排序(最后的元素
已经比父节点大了,会马上排好.
*/
}
//sort
template<class RandomAccessIterator>
void sort_heap(RandomAccessIterator first,
RandomAccessIterator last)
{
)
{
pop_heap(first, last--);
}
}
//结束后就是已经排序好的序列,就不是堆了
//make_heap
template<class RandomAccessIterator>
inline void make_heap(RandomAccessIterator first,
RandomAccessIterator last)
{
__make_heap(first, last, value_type(first), distance_type(first));
}
template<class RandomAccessIterator,
class T, class Distance>
void __make_heap(RandomAccessIterator first,
RandomAccessIterator last, T*,Distance*)
{
//判断边界情况,如果只有一个元素
直接返回
) return;
//len==元素个数
Distance len = last - first;
//parent ==
最后一个元素的父节点
Distance parent = (len -
) / ;
while (true)
{
//一直循环,直到全部排序好,从最后的三角形(最后的元素,父节点,左儿子(如果有的话))
//开始,直到根节点,就调整完毕了
__adjust_heap(first, parent, len, T(*(first + parent)));
)
return ;
parent--;
}
}
}
stl源码剖析 详细学习笔记heap的更多相关文章
- stl源码剖析 详细学习笔记 算法总览
//****************************基本算法***************************** /* stl算法总览,不在stl标准规格的sgi专属算法,都以 *加以标 ...
- stl源码剖析 详细学习笔记 hashtable
//---------------------------15/03/24---------------------------- //hashtable { /* 概述: sgi采用的是开链法完成h ...
- stl源码剖析 详细学习笔记 set map
// // set map.cpp // 笔记 // // Created by fam on 15/3/23. // // //---------------------------15/03 ...
- stl源码剖析 详细学习笔记 RB_tree (1)
// // RB_tree_STL.cpp // 笔记 // // Created by fam on 15/3/21. // // #include "RB_tree_STL.h&q ...
- stl源码剖析 详细学习笔记priority_queue slist
// // priority_queue.cpp // 笔记 // // Created by fam on 15/3/16. // // //------------------------- ...
- stl源码剖析 详细学习笔记 空间配置器
//---------------------------15/04/05---------------------------- /* 空间配置器概述: 1:new操作包含两个阶段操作 1>调 ...
- stl源码剖析 详细学习笔记 算法(1)
//---------------------------15/03/27---------------------------- //算法 { /* 质变算法:会改变操作对象之值 所有的stl算法都 ...
- stl源码剖析 详细学习笔记 RB_tree (2)
//---------------------------15/03/22---------------------------- //一直好奇KeyOfValue是什么,查了下就是一个和仿函数差不多 ...
- stl源码剖析 详细学习笔记stack queue
// // stack.cpp // 笔记 // // Created by fam on 15/3/15. // // //---------------------------15/03/1 ...
随机推荐
- kettle性能优化
普通开发电脑,如果没有网络查询步骤,kettle正常的速度应该在3000~20000条/秒.如果速度在2000条/秒一下,就可能需要调优. 性能优化的方式包括如下几种: 1.通过改变开始复制的数量(针 ...
- 【转】Java学习---Java核心数据结构(List,Map,Set)使用技巧与优化
[原文]https://www.toutiao.com/i6594587397101453827/ Java核心数据结构(List,Map,Set)使用技巧与优化 JDK提供了一组主要的数据结构实现, ...
- qt designer启动后不显示界面问题的原因与解决办法
Qt 5.6.1无论是在vs里双击ui文件还是直接启动designer.exe都一直无法显示界面,但任务管理器中可以看到该进程是存在的.前几天还正常的,但昨天加了一块NVIDIA的显卡(机器自带核显) ...
- MySQL基础之 AND和OR运算符
AND和OR运算符 作用:用于基于一个以上的条件对记录进行过滤 用法:可在WHERE子句中把两个或多个条件结合在一起. AND:如果第一个条件和第二个条件都成立,才会显示一条记录 OR:如果第一个条件 ...
- 乘风破浪:LeetCode真题_039_Combination Sum
乘风破浪:LeetCode真题_039_Combination Sum 一.前言 这一道题又是集合上面的问题,可以重复使用数字,来求得几个数之和等于目标. 二.Combination Sum ...
- 第 14 章 结构和其他数据形式(names)
*--------------------------------- names1.c -- 使用指向结构的指针 ---------------------------------*/ #includ ...
- phpstorm添加laravle语法支持
PHPStorm神器可以支持更友好的laravel框架代码提示,只需要执行如下才做: 第一步:在项目的composer.json中添加如下一行 "require": { " ...
- 团队-UML
UML设计 分工 刘双玉 李佳铭 杜宏庆 肖小强 汪志彬 江郑 符天愉 邓弘立 后台数据库 求购模块 浏览检索商品 即时聊天系统 商品管理 管理员系统 后台商品发布收藏系统 登录注册与个人信息系统 U ...
- Odoo Model内容详解
转载请注明原文地址:https://www.cnblogs.com/cnodoo/p/9390688.html 一:Odoo模型属性 1:_name 模型的唯一标识:如果没有继承其他模型 ...
- $\mathcal{Friends' \ \ Links}$友情链接
\(\mathcal{JuLao \ \& \ \ Dalao}\) \(\_rqy\) \(\_stdcall\) 并(吊)肩(锤)奋(死)斗(我)的\(Oier\) 王旭 苑骏康 张梓淳 ...