【数据结构】二叉树和堆

文章目录

  • 一、 什么是二叉树
  • 二、 二叉树的存储结构
    • 顺序存储视图
  • 三、 堆
    • 堆的结构及概念
    • 大堆和小堆
  • 四、 建堆
  • 五、 堆排序
  • 六、 topk问题

一、 什么是二叉树

二叉树,作为一种重要的数据结构,由节点组成,每个节点可以有两个子节点,通常称为左子节点和右子节点。二叉树是有序的,树中包含的各个节点的度不能超过2,即只能是0、1或者2。
在这里插入图片描述
特殊二叉树

  1. 满二叉树
    一个二叉树,如果每一个层的结点数都达到最大值,则这个二叉树就是满二叉树。也就是
    说,如果一个二叉树的层数为K,且结点总数是 2的k次方-1,则它就是满二叉树。
  2. 完全二叉树
    完全二叉树,作为一种效率很高的数据结构,是由满二叉树衍生出来的。一棵深度为K且有n个结点的二叉树,如果其每个节点都与深度为K的满二叉树中编号从1至n的节点一一对应,则这棵二叉树被称为完全二叉树。
    在这里插入图片描述

二、 二叉树的存储结构

二叉树一般可以使用两种结构存储,一种顺序结构,一种链式结构。

  1. 顺序存储
    顺序结构存储就是使用数组来存储,一般使用数组只适合表示完全二叉树, 因为不是完全二叉树会有空间的浪费。二叉树顺序存储在物理上是一个数组,在逻辑上是一颗二叉树。
    用数组存储的方式更方便查找根和子树。
  2. 链式存储
    二叉树的链式存储结构是指,用链表来表示一棵二叉树,即用链来指示元素的逻辑关系。 通常的方法是
    链表中每个结点由三个域组成,数据域和左右指针域,左右指针分别用来给出该结点左孩子和右孩子所在的链结点的存储地址 。

顺序存储视图

在这里插入图片描述

三、 堆

堆的结构及概念

如果有一个关键码的集合K ={ k0,k1,k2……,k(n-1)}【0,1,2,……,n-1这些都是下标】,把它的所有元素按完全二叉树的顺序存储方式存储在一个一维数组中,并满足:Ki<=k(2i+1)且Ki<=k2i+2【Ki>=k(2i+1)且Ki>=k(2i+2)】i=0,1,2…,则称为小堆【或大堆】。将根节点最大的堆叫做最大堆或大根堆,根节点最小的堆叫做最小堆或小根堆。

堆的性质:

  1. 堆中某个节点的值总是不大于或不小于其父节点的值;
  2. 堆总是一棵完全二叉树。

大堆和小堆

在这里插入图片描述
大堆:树中父亲的数据都大于等于孩子;
小堆:树中父亲的数据都小于等于孩子

四、 建堆

小堆为例
创建两个文件:
Heap.h

#pragma once
#include <stdio.h>
#include <stdlib.h>
#include <assert.h>
#include <stdbool.h>
typedef int HPDataType;
typedef struct Heap
{HPDataType* a;int size;int capacity;
}Heap;
//堆的初始化
void HeapInit(Heap* hp);
// 堆的销毁
void HeapDestory(Heap* hp);
// 堆的插入
void HeapPush(Heap* hp, HPDataType x);
// 堆的删除
void HeapPop(Heap* hp);
// 取堆顶的数据
HPDataType HeapTop(Heap* hp);
// 堆的数据个数
int HeapSize(Heap* hp);
// 堆的判空
bool HeapEmpty(Heap* hp);
// 对数组进行堆排序
void HeapSort(int* a, int n);
//交换位置
Swap(HPDataType* p1, HPDataType* p2);
//堆排序
void HeapSort(int* a, int n);
//向下交换
void AdjustDown(HPDataType* a, int n, int parent);
//向上查找
void Adjustup(HPDataType* a, HPDataType child);

Heap.c

#define   _CRT_SECURE_NO_WARNINGS 1
#include "Heap.h"
//堆的初始化
void HeapInit(Heap* hp)
{assert(hp);hp->a = NULL;hp->capacity = hp->size = 0;
}
// 堆的销毁
void HeapDestory(Heap* hp)
{assert(hp);free(hp->a);hp->a = NULL;hp->capacity = hp->size = 0;
}// 取堆顶的数据
HPDataType HeapTop(Heap* hp)
{assert(hp);assert(hp->size > 0);return hp->a[0];
}
// 堆的数据个数
int HeapSize(Heap* hp)
{assert(hp);return hp->size;
}
// 堆的判空
bool HeapEmpty(Heap* hp)
{assert(hp);return hp->size == 0;
}
//交换位置
Swap(HPDataType* p1, HPDataType* p2)
{HPDataType tmp = *p1;*p1 = *p2;*p2 = tmp;
}
//向上调整
Adjustup(HPDataType* a, HPDataType child)//从孩子位置向上调整
{//初始条件//中间过程//结束条件int parent = (child - 1) / 2;//可以画图得到这个公式while (child > 0){if (a[child] < a[parent])//小的往上换{Swap(&a[child], &a[parent]);child = parent;//继续往上调整parent = (child - 1) / 2;}else//父亲小于等于孩子{break;}}
}
// 堆的插入
void HeapPush(Heap* hp, HPDataType x)
{assert(hp);//扩容if (hp->capacity == hp->size){int newcapacity = hp->capacity == 0 ? 4 : hp->capacity * 2;HPDataType* tmp = (HPDataType*)realloc(hp->a, newcapacity * sizeof(HPDataType));if (tmp == NULL){perror("realloc fail!");exit(1);}hp->a = tmp;hp->capacity = newcapacity;}hp->a[hp->size] = x;hp->size++;Adjustup(hp->a, hp->size - 1);//插入完数据后向上调整
}
//向下调整
AdjustDown(HPDataType* a, int n, int parent)
{int child = parent * 2 + 1;//假设左孩子小while (child < n)//child >= n,说明孩子已经调整到叶子了{//防止越界访问,找到小的那个孩子再向下调if (child + 1 < n && a[child] > a[child + 1]){++child;}if (a[child] < a[parent]){Swap(&a[child], &a[parent]);parent = child;//继续往下调整child = parent * 2 + 1;}else{break;}}
}// 堆的删除
void HeapPop(Heap* hp)
{//Pop删除堆顶的数据(根位置)assert(hp);assert(hp->size > 0);//不能直接暴力删除,否则关系会乱套,兄弟变父子Swap(&hp->a[0], &hp->a[hp->size-1]);//第一个数据和最后一个数据交换hp->size--;AdjustDown(hp->a, hp->size, 0);//删除完需要用到向下调整算法
}

五、 堆排序

void HeapSort(int* a, int n)
{// 降序,建小堆// 升序,建大堆//建小堆for (int i = 1; i < n; i++){AdjustUp(a, i);}int end = n - 1;while (end > 0){Swap(&a[0], &a[end]);//将最小的数和最后的数进行交换AdjustDown(a, end, 0);--end;}
}
int main()
{int a[] = { 4,2,8,1,5,6,9,7 };HeapSort(a, sizeof(a) / sizeof(0));
}

调试更能理解这个过程
在这里插入图片描述

六、 topk问题

TOP-K问题:即求数据结合中前K个最大的元素或者最小的元素,一般情况下数据量都比较大。

比如:专业前10名、世界500强、富豪榜、游戏中前100的活跃玩家等。

对于Top-K问题,能想到的最简单直接的方式就是排序,但是:如果数据量非常大,排序就不太可取了(可能数据都不能一下子全部加载到内存中)。最佳的方式就是用堆来解决,基本思路如下:

假如我们现在想要找到100亿个整数中,取出前K个最大的数,如果我们直接对100亿个整数进行排序至少需要40G的内存,这会造成空间上巨大的浪费。

我们可以先取K个数建立一个小堆,再将后100亿-K个数依次与堆顶元素相比较,如果比堆顶元素大就将其替换后重新向下调整为一个小堆,再接着与下一个数相比,这样最终就可以找到前K个最大的整数了。(节约了大量的空间)

该方法的时间复杂度为:O(N*㏒⑵K)

空间复杂度为O(K)

![void CreateNDate()
{// 造数据int n = 100000;srand(time(0));const char* file = "F:\\vs2022\\二叉树_堆\\二叉树_堆\\data.txt";FILE* fin = fopen(file, "w");if (fin == NULL){perror("fopen error");return;}for (int i = 0; i < n; i++){int x = (rand() + i);fprintf(fin, "%d\n", x);}fclose(fin);
}void PrintTopK()
{int k = 0;printf("请输入要查找的前K个值>:");scanf("%d", &k);int* kminheap = (int*)malloc(sizeof(int) * k);if (kminheap == NULL){printf("malloc fali");return;}const char* file = "data.txt";FILE* fout = fopen(file, "r");if (fout == NULL){printf("fopen error");return;}//读取前k个数for (int i = 0; i < k; i++){fscanf(fout, "%d", &kminheap[i]);}//建堆for (int i = (k - 1 - 1) / 2; i >= 0; i--){AdjustDown(kminheap, k, i);}//读取剩下的N-K个数int x = 0;while (fscanf(fout, "%d", &x) > 0){if (x > kminheap[0]){kminheap[0] = x;}AdjustDown(kminheap, k, 0);}for (int i = 0; i < k; i++)\{printf("%d ", kminheap[i]);}printf("\n");
}

——————————————————————————————————————————
希望这篇博客对你有所帮助!!!
在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/diannao/16544.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

精品丨快速申请免费https证书

https域名证书对提高网站排名有一定的好处&#xff0c;所以当今很多企业为了给网站一个好的安全防护&#xff0c;就会去申请该证书。如今很多企业虽然重视网站的安全防护&#xff0c;但是也重视成本&#xff0c;所以为了节约成本会考虑申请免费的https证书。 第一个好处 企业不…

阻塞信号集和未决信号集_代码实现

1. 程序验证内容 将编号为0,1,2添加到阻塞信号集中&#xff0c;i<信号编号时&#xff0c;发出信号&#xff0c;观察未决信号集状态 当解除阻塞后&#xff0c;原先的信号是否执行&#xff0c;执行顺序是什么 2. 代码实现 #include <unistd.h> #include <stdlib.h…

Mac彩球一直转圈怎么解决 苹果电脑经常卡住 mac电脑运行慢怎么办

用Mac电脑办公的朋友在操作体积比较大的程序时&#xff0c;比如Final Cut Pro或者Photo Shop&#xff0c;一定遇到过鼠标变更彩球并不停转圈。这是怎么回事&#xff1f;今天本文就告诉大家Mac彩球一直转圈怎么解决&#xff0c;以及苹果电脑经常卡住是什么原因。 一、Mac彩球一直…

python03--元组-字典-集合

一、元组 tuple 数据形式 (数据1,数据2&#xff0c;数据3.....) 元组数据是不可修改&#xff0c;数据是有序的&#xff0c;支持下标获取数据 无法增加,修改&#xff0c;删除数据,只能进行数据查询 1-1 元组定义 # 元组定义 data_tuple1 (1,2,3,4) data_tuple2 (1.2,2.34,3.1…

Linux基础命令常见问题解决方案

Linux 基础命令常见问题解决方案 在Linux的日常使用中&#xff0c;用户经常会遇到各种各样的问题。本文旨在提供一个关于Linux基础命令的常见问题及其解决方案的全面指南。我们将覆盖30种不同的错误场景&#xff0c;并给出具体的解决步骤和示例&#xff0c;帮助初学者快速定位…

Wireshark 搜不到字符串?

一个原因是pcap里没有这个字符串&#xff0c; 另一个原因可能是ctrlF之后&#xff0c;选择搜索的地方不对&#xff0c;或者是编码方式选择的不对。 上面图片的第一个下拉框是要搜索的一个范围&#xff0c;是在哪一个panel搜索&#xff0c;范围说明在下面这个链接有详细说明&…

面试2. c++面试(领为军融)

1本段代码是否存在问题&#xff1f;. #include <iostream> void main() {for (unsigned short idx 0;idx <65535; idx){std::cout << idx <<std::endl;}std::cout<<"Execution completed!"<<std::endl; }解析&#xff1a; (1)可靠…

Java | Leetcode Java题解之第100题相同的树

题目&#xff1a; 题解&#xff1a; class Solution {public boolean isSameTree(TreeNode p, TreeNode q) {if (p null && q null) {return true;} else if (p null || q null) {return false;}Queue<TreeNode> queue1 new LinkedList<TreeNode>();…

【QGIS入门实战精品教程】10.6:QGIS制作酒店分布热力图

相关阅读: ArcGIS实验教程——实验四十二:ArcGIS密度分析(核密度、点密度、线密度) 【ArcGIS微课1000例】0086:基于七普人口数据的人口密度分析与制图 ArcGIS实验教程——实验二十四:人口密度制图 文章目录 一、加载酒店分布数据二、热力分析一、加载酒店分布数据 订阅专…

【数据结构与算法 | 基础篇】双向循环链表模拟双端队列

1. 前言 前文我们分别用链表&#xff0c;数组来实现了栈和队列. 而双端队列可以替代栈和队列并达到二者的效果.我们知道&#xff0c;栈的特点是只在栈顶操作元素&#xff0c;队列的特点是在队头pop元素&#xff0c;在队尾push元素. 而双端队列可以在队头和队尾分别进行pop与pu…

Python 新手最容易踩的坑

Python新手最容易踩的坑 缩进错误忘记引入模块使用未定义的变量不理解变量作用域字符串格式化错误乱用关键字多余的符号本期图书推荐&#xff1a;Python算法小讲堂---39个算法案例带你玩转Python内容简介获取方式 在学习 Python 的过程中&#xff0c;新手往往会遇到一些常见的陷…

如何用pyecharts工具制作动态变化柱状图

# # # 导入柱状图的包 # from pyecharts.charts import Bar # # # 创建一个柱状图 # bar1 Bar() # bar1.add_xaxis(["中国", "美国", "英国"]) # bar1.add_yaxis("1900年GDP总量", [10, 20, 30]) # # # 生成图 # bar1.render() # # #…

5月23日零钱兑换+组合总和Ⅳ

377.组合总和Ⅳ 给你一个由 不同 整数组成的数组 nums &#xff0c;和一个目标整数 target 。请你从 nums 中找出并返回总和为 target 的元素组合的个数。 题目数据保证答案符合 32 位整数范围。 示例 1&#xff1a; 输入&#xff1a;nums [1,2,3], target 4 输出&#xf…

全免费的数据恢复工具哪个好?分享2024年性价比超高的12款数据恢复软件!

当您丢失重要文件时&#xff0c;您应该可不想遇到措手不及的情况吧&#xff1f;相反&#xff0c;您需要在系统中使用一些可靠的数据恢复软件&#xff0c;但是全免费的数据恢复工具哪个好呢&#xff1f;别担心&#xff0c;本文将帮助您选择最适合您的解决方案。 如何挑选一款合适…

后端开发面试题3(附答案)

前言 在下首语言是golang,所以会用他作为示例。 原文参见 @arialdomartini的: Back-End Developer Interview Questions 设计模式相关问题 1. 请用一个例子表明,全局对象是邪恶的存在。 在Go语言中,虽然没有传统意义上的全局变量(全局对象),但可以通过包级别的变量来模…

【Rust日报】嵌入式 Rust:一份简化指南

EvilHelix 编辑器 EvilHelix 是一个采用 Vim 风格的模态编辑器&#xff0c;旨在提供快速且高效的编辑体验。它是 Helix 编辑器的一个分支&#xff0c;增加了 Vim binding&#xff0c;同时积极同步上游的特性&#xff0c;兼备了 Vim 和 Hexli 的优点&#xff1a; Vim 风格的模态…

blkio限制容器iops

/sys/fs/cgroup/blkio/blkio.throttle.read_iops_device /sys/fs/cgroup/blkio/blkio.throttle.write_iops_device lsblk 查看设备 kubectl get pod xxx -oyaml | grep -i id find / -name blkio.throttle.write_iops_device | grep id

栈和队列OJ题详解

一.有效的括号&#xff1a; 20. 有效的括号 - 力扣&#xff08;LeetCode&#xff09; 首先拿到这个题目&#xff0c;我的第一个思路是利用双指针来走&#xff0c;看看是不是匹配的 但是这种情况就把双指针的这个思路直接pass了&#xff0c;明明是匹配的括号&#xff0c;用双指…

1.3 Windows 的 CLion 开发环境安装

目录 1 C 语言的那些事 2 开发环境安装及新建项目 2.1 安装 MinGW 编译器 2.2 安装 CLion 开发环境

Android Audio基础——AudioFlinger回放录制线程(七)

AndioFlinger 作为 Android 的音频系统引擎,重任之一是负责输入输出流设备的管理及音频流数据的处理传输,这是由回放线程 PlaybackThread 及其派生的子类和录制线程 RecordThread 进行的。 一、基础介绍 1、关系图 ThreadBase:PlaybackThread 和 RecordThread 的基类。 Re…