我的编程空间,编程开发者的网络收藏夹
学习永远不晚

C语言如何解决二叉堆问题

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

C语言如何解决二叉堆问题

这篇文章给大家分享的是有关C语言如何解决二叉堆问题的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。

一、堆的概念

1、概述

堆是计算机科学中一类特殊的数据结构的统称。实现有很多,例如:大顶堆,小顶堆,斐波那契堆,左偏堆,斜堆 等等。从子结点个数上可以分为二叉堆,N叉堆等等。本文将介绍的是 二叉堆。

2、定义

二叉堆本质是一棵完全二叉树,所以每次元素的插入删除都能保证 O ( l o g 2 n ) O(log_2n) O(log2n)。根据堆的偏序规则,分为 小顶堆 和 大顶堆。小顶堆,顾名思义,根结点的关键字最小;大顶堆则相反。如图所示,表示的是一个大顶堆。

C语言如何解决二叉堆问题

3、性质

以大顶堆为例,它总是满足下列性质:
1)空树是一个大顶堆;

2)大顶堆中某个结点的关键字 小于等于 其父结点的关键字;

3)大顶堆是一棵完全二叉树。有关完全二叉树的内容,可以参考:画解完全二叉树。

如下图所示,任意一个从叶子结点到根结点的路径总是一个单调不降的序列。

C语言如何解决二叉堆问题

小顶堆只要把上文中的 小于等于 替换成 大于等于 即可。

4、作用

还是以大顶堆为例,堆能够在 O ( 1 ) O(1) O(1) 的时间内,获得 关键字 最大的元素。并且能够在 O ( l o g 2 n ) O(log_2n) O(log2n) 的时间内执行插入和删除。一般用来做 优先队列 的实现。

二、堆的存储结构

学习堆的过程中,我们能够学到一种新的表示形式。就是:利用 数组 来表示 链式结构。怎么理解这句话呢?

由于堆本身是一棵完全二叉树,所以我们可以把每个结点,按照层序映射到一个顺序存储的数组中,然后利用每个结点在数组中的下标,来确定结点之间的关系。

如图所示,描述的是堆结点下标和结点之间的关系,结点上的数字代表的是 数组下标。从左往右按照层序进行连续递增。

C语言如何解决二叉堆问题

1、根结点编号

根结点的编号,看作者的喜好。可以用 0 或者 1。本文的作者是 C语言 出身,所以更倾向于选择 0 作为根结点的编号(因为用 1 作为根结点编号的话,数组的第 0 个元素就浪费了)。

我们可以用一个宏定义来实现它的定义,如下:

#define root 0

2、孩子结点编号

那么,根结点的两个左右子树的编号,就分别为 1 和 2 了。以此类推,按照层序进行编号的话,1 的左右子树编号为 3 和 4;2 的左右子树编号为 5 和 6。

根据数学归纳法,对于编号为 i i i 的结点,它的左子树编号为 2 i + 1 2i+1 2i+1,右子树编号为 2 i + 2 2i+2 2i+2。用宏定义实现如下:

#define lson(idx) (2*idx+1)#define rson(idx) (2*idx+2)

由于这里涉及到乘 2,所以我们还可以用左移位运算来优化乘法运算,如下:

#define lson(idx) (idx << 1|1)#define rson(idx) ((idx + 1) << 1)

3、父结点编号

同样,父结点编号也可以通过数学归纳法得出,当结点编号为 i i i 时,它的父结点编号为 i &minus; 1 2 \frac {i-1} {2} 2i&minus;1,利用C语言实现如下:

#define parent(idx) ((idx - 1) / 2)

这里涉及到除 2,可以利用右移运算符进行优化,如下:

#define parent(idx) ((idx - 1) >> 1)

这里利用补码的性质,根结点的父结点得到的值为 -1;

4、数据域

堆数据元素的数据域可以定义两个:关键字 和 值,其中关键字一般是整数,方便进行比较确定大小关系;值则是用于展示用,可以是任意类型,可以用typedef struct进行定义如下:

typedef struct {    int key;      // (1)    void *any;    // (2)}DataType;
  • (1) 关键字;

  • (2) 值,定义成一个空指针,可以用来表示任意类型;

5、堆的数据结构

由于堆本质上是一棵完全二叉树,所以将它一一映射到数组后,一定是连续的。我们可以用一个数组来代表一个堆,在C语言中的数组拥有一个固定长度,可以用一个Heap结构体表示如下:

typedef struct {    DataType *data;  // (1)    int size;        // (2)    int capacity;    // (3)}Heap;
  • (1) 堆元素所在数组的首地址;

  • (2) 堆元素个数;

  • (3) 堆的最大元素个数;

三、堆的常用接口

1、元素比较

两个堆元素的比较可以采用一个比较函数compareData来完成,比较过程就是对关键字key进行比较的过程,以大顶堆为例:

a. 大于返回 -1,代表需要执行交换;

b. 小于返回 1,代表需要执行交换;

c. 等于返回 0,代表需要执行交换;

int compareData(const DataType* a, const DataType* b) {    if(a->key > b->key) {        return -1;    }else if(a->key < b->key) {        return 1;    }    return 0;}

2、交换元素

交换两个元素的位置,也是堆这种数据结构中很常见的操作,C语言实现也比较简单,如下:

void swap(DataType* a, DataType* b) {    DataType tmp = *a;    *a = *b;    *b = tmp;}

3、空判定

空判定是一个查询接口,即询问堆是否是空的,实现如下:

bool HeapIsEmpty(Heap *heap) {    return heap->size == 0;}

4、满判定

满判定是一个查询接口,即询问堆是否是满的,实现如下:

bool heapIsFull(Heap *heap) {    return heap->size == heap->capacity;}

5、上浮操作

对于大顶堆而言,从它叶子结点到根结点的元素关键字一定是单调不降的,如果某个元素出现了比它的父结点大的情况,就需要进行上浮操作。

上浮操作就是对 当前结点 和 父结点 进行比较,如果它的关键字比父结点大(compareData返回-1的情况),将它和父结点进行交换,继续上浮操作;否则,终止上浮操作。

如图所示,代表的是一个关键字为 95 的结点,通过不断上浮,到达根结点的过程。上浮完毕以后,它还是一个大顶堆。

C语言如何解决二叉堆问题

上浮过程的 C语言 实现如下:

void heapShiftUp(Heap* heap, int curr) {               // (1)    int par = parent(curr);                            // (2)    while(par >= root) {                               // (3)        if( compareData( &heap->data[curr], &heap->data[par] ) < 0 ) {            swap(&heap->data[curr], &heap->data[par]); // (4)             curr = par;            par = parent(curr);        }else {            break;                                     // (5)         }    }}

(1) heapShiftUp这个接口是一个内部接口,所以用小写驼峰区分,用于实现对堆中元素进行插入的时候的上浮操作;

(2) curr表示需要进行上浮操作的结点在堆中的编号,par表示curr的父结点编号;

(3) 如果已经是根结点,则无须进行上浮操作;

(4) 子结点的关键字 大于 父结点的关键字,则执行交换,并且更新新的 当前结点 和 父结点编号;

(5) 否则,说明已经正确归位,上浮操作结束,跳出循环;

6、下沉操作

对于大顶堆而言,从它 根结点 到 叶子结点 的元素关键字一定是单调不增的,如果某个元素出现了比它的某个子结点小的情况,就需要进行下沉操作。

下沉操作就是对 当前结点 和 关键字相对较小的子结点 进行比较,如果它的关键字比子结点小,将它和这个子结点进行交换,继续下沉操作;否则,终止下沉操作。

如图所示,代表的是一个关键字为 19 的结点,通过不断下沉,到达叶子结点的过程。下沉完毕以后,它还是一个大顶堆。

C语言如何解决二叉堆问题

下沉过程的 C语言 实现如下:

void heapShiftDown(Heap* heap, int curr) {            // (1)    int son = lson(curr);                             // (2)    while(son < heap->size) {        if( rson(curr) < heap->size ) {            if( compareData( &heap->data[rson(curr)], &heap->data[son] ) < 0 ) {                son = rson(curr);                     // (3)             }                }        if( compareData( &heap->data[son], &heap->data[curr] ) < 0 ) {            swap(&heap->data[son], &heap->data[curr]); // (4)            curr = son;            son = lson(curr);        }else {            break;                                     // (5)         }    }}void heapShiftDown(Heap* heap, int curr) {            // (1)    int son = lson(curr);                             // (2)    while(son < heap->size) {        if( rson(curr) < heap->size ) {            if( compareData( &heap->data[rson(curr)], &heap->data[son] ) < 0 ) {                son = rson(curr);                     // (3)             }                }        if( compareData( &heap->data[son], &heap->data[curr] ) < 0 ) {            swap(&heap->data[son], &heap->data[curr]); // (4)            curr = son;            son = lson(curr);        }else {            break;                                     // (5)         }    }}

(1) heapShiftDown这个接口是一个内部接口,所以用小写驼峰区分,用于对堆中元素进行删除的时候的下沉调整;

(2) curr表示需要进行下沉操作的结点在堆中的编号,son表示curr的左儿子结点编号;

(3) 始终选择关键字更小的子结点;

(4) 子结点的值小于父结点,则执行交换;

(5) 否则,说明已经正确归位,下沉操作结束,跳出循环;

四、堆的创建

1、算法描述

通过给定的数据集合,创建堆。可以先创建堆数组的内存空间,然后一个一个执行堆的插入操作。插入操作的具体实现,会在下文继续讲解。

2、动画演示

C语言如何解决二叉堆问题

3、源码详解

Heap* HeapCreate(DataType *data, int dataSize, int maxSize) {    // (1)    int i;    Heap *h = (Heap *)malloc( sizeof(Heap) );                    // (2)    h->data = (DataType *)malloc( sizeof(DataType) * maxSize );  // (3)    h->size = 0;                                                 // (4)    h->capacity = maxSize;                                       // (5)    for(i = 0; i < dataSize; ++i) {        HeapPush(h, data[i]);                                    // (6)    }    return h;                                                    // (7)}

(1) 给定一个元素个数为dataSize的数组data,创建一个最大元素个数为maxSize的堆并返回堆的结构体指针;

(2) 利用malloc申请堆的结构体的内存;

(3) 利用malloc申请存储堆数据的数组的内存空间;

(4) 初始化空堆;

(5) 初始化堆最大元素个数为maxSize

(6) 遍历数组执行堆的插入操作,插入的具体实现HeapPush接下来会讲到;

(7) 最后,返回堆的结构体指针;

五、堆元素的插入

1、算法描述

堆元素的插入过程,就是先将元素插入堆数组的最后一个位置,然后执行上浮操作;

2、动画演示

C语言如何解决二叉堆问题

3、源码详解

bool HeapPop(Heap *heap) {    if(HeapIsEmpty(heap)) {        return false;                               // (1)    }    heap->data[root] = heap->data[ --heap->size ];  // (2)    heapShiftDown(heap, root);                      // (3)    return true;}

(1) 堆已满,不能进行插入;

(2) 插入堆数组的最后一个位置;

(3) 对最后一个位置的 堆元素 执行上浮操作;

五、堆元素的删除

1、算法描述

堆元素的删除,只能对堆顶元素进行操作,可以将数组的最后一个元素放到堆顶,然后对堆顶元素进行下沉操作。

2、动画演示

C语言如何解决二叉堆问题

3、源码详解

bool HeapPop(Heap *heap) {    if(HeapIsEmpty(heap)) {        return false;                               // (1)    }    heap->data[root] = heap->data[ --heap->size ];  // (2)    heapShiftDown(heap, root);                      // (3)    return true;}
  • (1) 堆已空,无法执行删除;

  • (2) 将堆数组的最后一个元素放入堆顶,相当于删除了堆顶元素;

  • (3) 对堆顶元素执行下沉操作;

感谢各位的阅读!关于“C语言如何解决二叉堆问题”这篇文章就分享到这里了,希望以上内容可以对大家有一定的帮助,让大家可以学到更多知识,如果觉得文章不错,可以把它分享出去让更多的人看到吧!

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

C语言如何解决二叉堆问题

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

C语言如何解决二叉堆问题

这篇文章给大家分享的是有关C语言如何解决二叉堆问题的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。一、堆的概念1、概述堆是计算机科学中一类特殊的数据结构的统称。实现有很多,例如:大顶堆,小顶堆,斐波那契堆,左偏堆,
2023-06-26

C语言平衡二叉树问题怎么解决

这篇文章主要介绍“C语言平衡二叉树问题怎么解决”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“C语言平衡二叉树问题怎么解决”文章能帮助大家解决问题。一、题目描述给定一个二叉树,判断它是否是高度平衡的二
2023-06-30

C语言怎么用堆解决Topk问题

这篇文章给大家分享的是有关C语言怎么用堆解决Topk问题的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。前言将详细讲解如何利用小根堆的方法解决TopK问题,这么多数据要处理,该算法时间复度居然只需TopK问题Top
2023-06-21

Java语言如何实现二叉堆的打印

这篇文章将为大家详细讲解有关Java语言如何实现二叉堆的打印,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。二叉堆是一种特殊的堆,二叉堆是完全二元树(二叉树)或者是近似完全二元树(二叉树)。二叉堆有两种:最
2023-05-30

C语言堆排序经典算法TopK问题怎么解决

本文小编为大家详细介绍“C语言堆排序经典算法TopK问题怎么解决”,内容详细,步骤清晰,细节处理妥当,希望这篇“C语言堆排序经典算法TopK问题怎么解决”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知识吧。问题描述:从a
2023-07-05

如何解决C语言中for循环问题

本篇内容主要讲解“如何解决C语言中for循环问题”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“如何解决C语言中for循环问题”吧!今天分享一下C语言中的for循环中我们常常忽略的小问题。举一个小
2023-06-07

c语言内存溢出问题如何解决

C语言内存溢出问题可以通过以下几种方式来解决:1. 检查代码逻辑:检查代码中的循环、递归、动态内存分配等地方是否存在错误,比如没有正确释放内存或者使用了未初始化的指针。2. 动态内存管理:在使用动态内存分配函数(如malloc、calloc
2023-10-10

如何使用C语言解决八皇后问题

这篇文章主要讲解了“如何使用C语言解决八皇后问题”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“如何使用C语言解决八皇后问题”吧!前言八皇后问题是一个古老而著名的问题。该问题是19世纪著名的数
2023-06-08

c语言如何构建一个静态二叉树

这篇文章主要介绍“c语言如何构建一个静态二叉树”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“c语言如何构建一个静态二叉树”文章能帮助大家解决问题。第一、树的构建定义树结构struct BTNode
2023-06-16

C语言堆排序经典算法TopK问题解析

这篇文章主要为大家介绍了C语言堆排序经典算法TopK问题解析,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
2023-05-15

C语言如何解决青蛙跳台阶问题

小编给大家分享一下C语言如何解决青蛙跳台阶问题,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!1. 基础问题题目描述一只青蛙一次可以跳上 1 级台阶,也可以跳上 2
2023-06-29

C语言中如何实现二叉树的后序遍历

小编给大家分享一下C语言中如何实现二叉树的后序遍历,希望大家阅读完这篇文章之后都有所收获,下面让我们一起去探讨吧!首先我们从两个方面讲解二叉树的后序遍历(递归+迭代)一.二叉树的后序遍历.(递归)思想:首先我们从二叉树的根节点开始先遍历其左
2023-06-29

C语言算法中如何解决佩奇借书问题

小编给大家分享一下C语言算法中如何解决佩奇借书问题,希望大家阅读完这篇文章之后都有所收获,下面让我们一起去探讨吧!1. 问题描述佩奇有5本新书,要借给A、B、C这3位小朋友,若每人每次只能借1本,则可以有多少种不同的借法?2. 题目分析本题
2023-06-29

C语言中如何利用递归实现线索二叉树

这篇“C语言中如何利用递归实现线索二叉树”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“C语言中如何利用递归实现线索二叉树”文
2023-06-17

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录