【堆】建堆、插入、删除、堆排序

2023-05-16

参考

堆就是利用数组来实现二叉树,可用于构建优先队列、堆排序、TopK问题等。可分为:

  • 最大堆:父节点的值比其子节点大
  • 最小堆:父节点的值比其子节点小

堆的根节点存放了最小(或最大)元素,但是其它节点的排序是未知的,即左节点不一定比右节点大(或小),堆的搜索相对于二叉搜索树更慢,二叉搜索树(如AVL)是平衡二叉树,搜索时间复杂度为O(nlogn),而堆的目的只是为了获得最大(或最小)元素,将其存储在头节点。

实现堆只需要一个顺序存储结构的数组,相对于二叉树而言,空间开销更低,二叉树还需要分配左右指针占用内存更多。

假设父节点的位置为i(节点下标从0开始),则其左孩子位置为2i+1,右孩子位置为2i+2,这种存储方式与完全二叉树使用数组存储是一样的。
在这里插入图片描述

1. 建立最大堆
创建堆数组,将给定数组拷贝给堆数组;调整堆,从最后一个非叶节点开始进行shiftdown下滑操作,如下图,从97开始往前调整堆,将当前节点与其孩子节点进行比较,若孩子节点大于父节点,则将父节点下滑,孩子节点上浮
在这里插入图片描述
97和65均比孩子节点大不需下浮,节点38比孩子节点97、76都要小,将其与较大的97节点进行交换
在这里插入图片描述
交换后需要进行递归,直到下浮的38节点比孩子节点小
在这里插入图片描述
最后38节点被交换到了叶子节点
在这里插入图片描述
建堆完成后如下
在这里插入图片描述
2.插入新节点
在建立最大堆后,插入节点有可能破坏堆的结构,插入的节点被放到堆的尾部,如下,插入节点85,此时85比父节点49大,不满足最大堆结构
在这里插入图片描述
此时需要沿着以下路径将节点85上浮shiftup操作
在这里插入图片描述
最后85节点调整到如下位置
在这里插入图片描述
3.删除根节点
如下删除根节点97
在这里插入图片描述
将尾部节点替换到根节点,然后从根节点开始进行shiftdown下滑操作
在这里插入图片描述
调整后堆的结构如下,节点49被下浮到第3层
在这里插入图片描述

4.JAVA实现

class maxHeap {
	private int MAX_SIZE, NOW_SIZE, HEAP[];

	public maxHeap(int nums[]) {
		/*
		 * Create a heap using nums[]
		 */
		MAX_SIZE = 100;
		HEAP = new int[MAX_SIZE];
		for (int i = 0; i < nums.length; i++) {
			HEAP[i] = nums[i];
			NOW_SIZE++;
		}
		for (int i = NOW_SIZE / 2 - 1; i >= 0; i--) {
			shiftdown(i);
		}
	}

	public void shiftdown(int ind) {
		/*
		 * Sink the node to satisify the HEAP
		 */
		int left_child = 2 * ind + 1;
		int right_child = 2 * ind + 2;
		int large_ind = ind;
		if (left_child < NOW_SIZE && HEAP[large_ind] < HEAP[left_child]) {
			large_ind = left_child;
		}
		if (right_child < NOW_SIZE && HEAP[large_ind] < HEAP[right_child]) {
			large_ind = right_child;
		}
		if (large_ind != ind) {
			/*
			 * exchange the father node and child node then call shiftdown again
			 */
			int temp = HEAP[large_ind];
			HEAP[large_ind] = HEAP[ind];
			HEAP[ind] = temp;
			shiftdown(large_ind);
		}
	}

	public void shiftup(int ind) {
		/*
		 * when insert a value, need to shiftup the node
		 */
		// parent = (ind - 1) / 2
		int parent = (int) (ind - 1) / 2;
		if (parent >= 0 && HEAP[parent] < HEAP[ind]) {
			int temp = HEAP[parent];
			HEAP[parent] = HEAP[ind];
			HEAP[ind] = temp;
			shiftup(parent);
		}
	}

	public void insert(int val) {
		if (NOW_SIZE < MAX_SIZE - 1) {
			HEAP[NOW_SIZE] = val;
			shiftup(NOW_SIZE);
			NOW_SIZE++;
		}
	}

	public void delete() {
		HEAP[0] = HEAP[NOW_SIZE - 1];
		NOW_SIZE--;
		shiftdown(0);
	}

	public void printHeap() {
		for (int i = 0; i < NOW_SIZE; i++)
			System.out.print(HEAP[i] + " ");
		System.out.println();
	}
}

// 测试
public class main {
	public static void main(String[] args) {
		int nums[] = { 1, 2, 7, 9, 3, 8, 5, 6, 7 };
		maxHeap heap = new maxHeap(nums);
		heap.printHeap();

		heap.insert(10);
		heap.printHeap();

		heap.insert(5);
		heap.printHeap();

		heap.delete();
		heap.printHeap();

		heap.delete();
		heap.printHeap();

	}
}

5.堆排序
在建好堆后,可以进一步实现堆排序,由于堆顶始终是最小(或最大)元素,因此每次取出堆顶元素然后将末尾元素替换到堆顶,并从堆顶进行shiftdown下浮操作调堆,即可完成堆排序。

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

【堆】建堆、插入、删除、堆排序 的相关文章

  • 【PIL】将数组转换为图像并保存

    参考自 span class token keyword from span PIL span class token keyword import span Image im span class token operator 61 sp
  • 【python】join连接字符串

    join函数用于将序列中的元素以指定字符串连接在一起 xff0c 例如 span class token builtin str span span class token operator 61 span span class token
  • 【python】复制文件

    span class token keyword import span shutil shutil span class token punctuation span copyfile span class token punctuati
  • 【pandas】merge和join

    merge和join都是将不同的dataframe进行合并 xff0c 区别在于merge是通过key将frame合并 xff0c 而join通过index进行合并 merge 两个frame必须要有同一种属性可以作为键值 xff0c 如下
  • 【python】画图保存为emf

    emf为矢量图 xff0c 是office支持的格式 xff0c python不能直接保存为emf xff0c 首先保存为svg格式 xff0c 如下 span class token keyword import span matplot
  • 【seaborn】绘制概率密度估计图

    span class token keyword import span seaborn span class token keyword as span sns plt span class token punctuation span
  • 【C++】Socket通信例子

    创建两个工程文件 xff0c Server和Client 服务器模板代码 span class token macro property span class token directive keyword include span spa
  • 【STL】map基本用法

    C 43 43 中的map类似于python中的字典 xff0c 形如 lt key value gt 对 创建map对象 可以像python中的字典一样直接使用IDMPA key 获取对应的值 span class token macro
  • 【string】字符串拷贝strcpy

    strcpy即string copy span class token keyword char span span class token operator span str span class token operator 61 sp
  • web控件开发

    https www bbsmax com A gGdX6v11J4 https open chrome 360 cn extension dev overview html 扩展示例 Mozilla MDN https edge micro
  • 【string】获取字符串长度strlen

    span class token keyword char span str span class token punctuation span span class token punctuation span span class to
  • 【string】int转string to_string

    使用to string函数将整型变量转为字符串 span class token keyword int span n span class token operator 61 span span class token number 10
  • 【C++】string、char*互相转换

    string 2 char c str 方法返回一个const char 类型的指针变量 xff0c 使用strcpy函数copy string str span class token operator 61 span span clas
  • 【string】字符串分割strock

    使用strock将字符串按特定分隔符分割 span class token macro property span class token directive keyword include span span class token st
  • 【string】字符串转int stoi

    使用stoi函数将字符串转为int型 xff0c 需要 include lt string gt span class token keyword char span chs span class token punctuation spa
  • Visual Studio解决const char *与LPCWSTR 不兼容

    项目 gt 属性 gt 配置属性 gt 高级 xff0c 将字符集改为未设置
  • 【C++】文件读写指针定位

    主要函数 xff1a 指针定位函数SetFilePointer xff0c 读文件ReadFile xff0c 写文件WriteFile 首先使用CreateFile创建文件 xff0c SetFilePointer函数将指针定位到文件指定
  • 【C++】程序计时

    span class token macro property span class token directive keyword include span span class token string lt iostream gt s
  • 【双指针】15.三数之和

    题目 给你一个包含 n 个整数的数组 nums xff0c 判断 nums 中是否存在三个元素 a xff0c b xff0c c xff0c 使得 a 43 b 43 c 61 0 xff1f 请你找出所有满足条件且不重复的三元组 注意
  • 函数定义中的冒号和箭头

    函数中变量后面加冒号和类型表示该参数的建议类型 xff0c 如下参数A的建议类型是int xff0c 参数B的建议类型是str xff0c gt 表示该函数的返回值类型 xff0c 例如fun函数的返回值类型是str span class

随机推荐

  • chrom控件命令行加载

    34 C Users lt name gt AppData Local Google Chrome Application chrome exe 34 load extension 61 34 lt path to unpacked ext
  • 【二叉树】创建、先序遍历、中序遍历、后序遍历、层序遍历 Java实现

    二叉树的创建 二叉树的创建可以采用递归方式 xff0c 传入一个数组 xff0c 例如数组 3 9 20 null null 15 7 表示的二叉为 span class token number 3 span span class tok
  • 【二叉树】平衡二叉树

    平衡二叉树是一种二叉查找树又称为AVL树 Adelsen Velskii and Landis xff0c 特点为每个节点的左 右子树深度之差的绝对值不大于1 xff0c 左子树的值小于右子树 xff0c 重要操作为插入和删除 插入 插入新
  • win10搭建hadoop2.7.7

    配置java环境 配置java环境 xff0c 官网下载jdk较慢 xff0c 百度网盘 xff1a 链接 xff1a https pan baidu com s 1wX7LxPMjcS9QGc4c4cPJgw 提取码 xff1a 9e38
  • win10配置spark

    下载spark压缩包 xff0c 链接 xff1a https pan baidu com s 1y5JlMdtkrZFyTJWKtuuZ Q 提取码 xff1a z64y 解压tar gz文件 配置环境变量 xff0c 系统变量Path中
  • 【HDFS】上传、查看、下载、删除文件命令

    上传 首先启动HDFS xff0c 任意目录下输入命令start dfs xff08 若没有配置sbin的环境变量则需要在sbin目录下打开cmd输入该命令 xff09 xff0c 出现以下两个框框 在需要上传文件的文件路径下打开cmd命令
  • IntelliJ IDEA开发spark应用(scala)

    配置spark环境 xff0c 可参考官网下载 IntelliJ IDEA xff0c 然后安装 xff0c 一直next即可 安装Scala插件 创建一个新工程 Ctrl 43 Shift 43 Alt 43 s xff0c 导入spar
  • 【STL】vector简单使用

    参考 需要头文件 span class token macro property span class token directive keyword include span span class token string lt iost
  • 【python效率优化】使用map优化for循环

    python提供的高级函数map将一个函数作用于可迭代对象的每一个元素 xff0c 底层自动实现并行 xff0c 运行速度比for循环要快 xff0c 对于无前后联系的for循环 xff0c 可以使用map进行优化 xff0c 以下例子对比
  • 【数组】121. 买卖股票的最佳时机

    题目 给定一个数组 xff0c 它的第 i 个元素是一支给定股票第 i 天的价格 如果你最多只允许完成一笔交易 xff08 即买入和卖出一支股票一次 xff09 xff0c 设计一个算法来计算你所能获取的最大利润 注意 xff1a 你不能在
  • 【双指针】26. 删除排序数组中的重复项

    题目 给定一个排序数组 xff0c 你需要在 原地 删除重复出现的元素 xff0c 使得每个元素只出现一次 xff0c 返回移除后数组的新长度 不要使用额外的数组空间 xff0c 你必须在 原地 修改输入数组 并在使用 O 1 额外空间的条
  • centos下安装chrome

    到网页 https www google cn chrome 点击安装 下载 rpm安装包 安装即可 root 64 localhost 下载 yum localinstall google chrome stable current x8
  • 【双指针】80. 删除排序数组中的重复项 II

    题目 给定一个排序数组 xff0c 你需要在原地删除重复出现的元素 xff0c 使得每个元素最多出现两次 xff0c 返回移除后数组的新长度 不要使用额外的数组空间 xff0c 你必须在原地修改输入数组并在使用 O 1 额外空间的条件下完成
  • 【双指针】27. 移除元素

    题目 给你一个数组 nums 和一个值 val xff0c 你需要 原地 移除所有数值等于 val 的元素 xff0c 并返回移除后数组的新长度 不要使用额外的数组空间 xff0c 你必须仅使用 O 1 额外空间并 原地 修改输入数组 元素
  • 【栈】155. 最小栈

    题目 设计一个支持 push xff0c pop xff0c top 操作 xff0c 并能在常数时间内检索到最小元素的栈 push x 将元素 x 推入栈中 pop 删除栈顶的元素 top 获取栈顶元素 getMin 检索栈中的最小元素
  • 【数组】初始化、获取长度

    初始化 xff0c 获取长度 span class token keyword public span span class token keyword class span span class token class name main
  • 【Stack】简单使用

    入栈 xff1a add获取栈顶元素 xff1a peek出栈 xff1a pop span class token keyword import span java span class token punctuation span ut
  • 【HashMap】基本操作

    添加键值对put获取key对应的value get遍历 xff1a keySet span class token keyword import span java span class token punctuation span uti
  • 【单调栈】496. 下一个更大元素 I

    题目 给定两个 没有重复元素 的数组 nums1 和 nums2 xff0c 其中nums1 是 nums2 的子集 找到 nums1 中每个元素在 nums2 中的下一个比其大的值 nums1 中数字 x 的下一个更大元素是指 x 在 n
  • 【堆】建堆、插入、删除、堆排序

    参考 堆就是利用数组来实现二叉树 xff0c 可用于构建优先队列 堆排序 TopK问题等 可分为 xff1a 最大堆 xff1a 父节点的值比其子节点大最小堆 xff1a 父节点的值比其子节点小 堆的根节点存放了最小 xff08 或最大 x