Heapsort排序思路

将整个数组看作一个二叉树heap, 下标0为堆顶层, 下标1, 2为次顶层, 然后每层就是"3,4,5,6", "7, 8, 9, 10, 11, 12, 13, 14", ..., 对于其中的每一个非叶子节点, 其子节点的下标为 2 * pos + 1 和 2 * pos + 2

循环进行以下工作:

  1. 标记当前堆的右边界
  2. 将堆初始化为大顶堆
  3. 将堆顶置换到右边界, 同时将右边界左移一位(即将堆缩小一格)

将堆初始化为大顶堆的处理

  1. 从最后一个非叶子节点开始, 其下标为  length / 2 - 1, length为当前的堆大小, 往前挨个处理
  2. 检查节点值, 对节点与其子节点的值大小i进行比较, 如果子节点值比当前节点大, 将最大值的那个子节点换上来
  3. 如果发生了交换, 那么要对发生交换的这个子节点位置也进行检查, 如果还有交换, 则继续往下查, 直到达到叶子节点
public class DemoHeapSort {
// 待排序的数组
private int[] ints;
// 当前堆的右边界(下标)
private int limit; public DemoHeapSort(int ... ints) {
this.ints = ints;
this.limit = ints.length - 1;
} public void sortAll() {
while (limit > 0) {
resort();
swap(0, limit);
limit--;
}
} public void resort() {
// 起点pos为当前堆length / 2 - 1
for (int i = (limit + 1) / 2 - 1; i >= 0; i--) {
int pos = i;
while (true) {
pos = check(pos);
if (pos == 0) {
break;
}
}
}
} /**
* 检查pos位置, 自上而下判断是否都满足大顶堆的要求
* 其左右子节点的下标为 2*pos+1 和 2*pos+2
*
* @param pos 当前要检查的下标
* @return 如果无调整, 则返回0, 如果有调整, 则返回被调整的子节点下标
*/
private int check(int pos) {
int posLeft = 2 * pos + 1;
int posRight = 2 * pos + 2;
if (posLeft > limit) { // 没有子节点
return 0;
}
if (posRight > limit) { // 仅有左节点
if (ints[pos] < ints[posLeft]) {
swap(pos, posLeft);
return posLeft;
} else {
return 0;
}
}
// 左右节点都有
if (ints[posLeft] > ints[posRight]) {
if (ints[pos] < ints[posLeft]) {
swap(pos, posLeft);
return posLeft;
} else {
return 0;
}
} else {
if (ints[pos] < ints[posRight]) {
swap(pos, posRight);
return posRight;
} else {
return 0;
}
}
} public void print() {
for (int a : ints) {
System.out.printf("%3d", a);
}
System.out.print("\n");
} private void swap(int a, int b) {
int tmp = ints[a];
ints[a] = ints[b];
ints[b] = tmp;
} public static void main(String[] args) {
DemoHeapSort hs = new DemoHeapSort(10, 97, 9, 1,63, 64, 8, 17, 33, 7, 21, 0, 7, 75, 13, 18, 2, 99, 87);
hs.sortAll();
hs.print();
}
}

C代码, 在原数组上排序

void heap_sort(int *nums, int size) {
for (int i = 0; i < size; i++) {
int num = *(nums + i);
printf("%d ", num);
}
printf("\n");
for (int i = 0; i < size; i++) {
int length = size - i;
for (int pos = length / 2 - 1; pos >= 0; pos-- ) {
int p = pos;
do {
p = check(nums + i, length - 1, p);
} while (p != 0);
}
for (int j = 0; j < size; j++) {
int num = *(nums + j);
printf("%d ", num);
}
printf(" %d\n", i);
}
} int check(int *nums, int limit, int pos) {
int ls = pos * 2 + 1;
int rs = pos * 2 + 2;
if (ls > limit) {
return 0;
}
int p = *(nums + pos);
if (rs > limit) {
int l = *(nums + ls);
if (l > p) {
swap(nums + ls, nums + pos);
return ls;
} else {
return 0;
}
} else {
int l = *(nums + ls);
int r = *(nums + rs);
if (l > r) {
if (l > p) {
swap(nums + ls, nums + pos);
return ls;
} else {
return 0;
}
} else {
if (r > p) {
swap(nums + rs, nums + pos);
return rs;
} else {
return 0;
}
}
}
} void swap(int *a, int *b) {
int t = *a;
*a = *b;
*b = t;
}

堆排序Heapsort的Java和C代码的更多相关文章

  1. 数据结构 - 堆排序(heap sort) 具体解释 及 代码(C++)

    堆排序(heap sort) 具体解释 及 代码(C++) 本文地址: http://blog.csdn.net/caroline_wendy 堆排序包括两个步骤: 第一步: 是建立大顶堆(从大到小排 ...

  2. 正则表达式学习笔记(附:Java版示例代码)

    具体学习推荐:正则表达式30分钟入门教程 .         除换行符以外的任意字符\w      word,正常字符,可以当做变量名的,字母.数字.下划线.汉字\s        space,空白符 ...

  3. java俄罗斯方块游戏代码

    java俄罗斯方块游戏代码: package com; import java.awt.Color; import java.awt.Graphics; import java.awt.event.K ...

  4. java常用用代码

    /** *Java获取IP代码 */ import java.awt.GridLayout; import java.awt.event.ActionEvent; import java.awt.ev ...

  5. java学用代码

    /** *Java获取IP代码 */ import java.awt.GridLayout; import java.awt.event.ActionEvent; import java.awt.ev ...

  6. 分享非常有用的Java程序(关键代码)(七)---抓屏程序

    原文:分享非常有用的Java程序(关键代码)(七)---抓屏程序 import java.awt.Dimension; import java.awt.Rectangle; import java.a ...

  7. 分享非常有用的Java程序 (关键代码)(六)---解析/读取XML 文件(重要)

    原文:分享非常有用的Java程序 (关键代码)(六)---解析/读取XML 文件(重要) XML文件 <?xml version="1.0"?> <student ...

  8. 分享非常有用的Java程序 (关键代码)(五)---把 Array 转换成 Map

    原文:分享非常有用的Java程序 (关键代码)(五)---把 Array 转换成 Map import java.util.Map; import org.apache.commons.lang.Ar ...

  9. 分享非常有用的Java程序 (关键代码)(四)---动态改变数组的大小

    原文:分享非常有用的Java程序 (关键代码)(四)---动态改变数组的大小 /** * Reallocates an array with a new size, and copies the co ...

随机推荐

  1. Java使用JsonPatch

    老规矩,概念的东西不再此处体现,baidu即可自行解决,直入主题,动手第一. 导入所需的jar文件 pom.xml     <dependencies>        <depend ...

  2. 发布WS接口与实现WS接口[小列子]

    webservice简介:Web Service技术, 能使得运行在不同机器上的不同应用无须借助附加的.专门的第三方软件或硬件, 就可相互交换数据或集成.依据Web Service规范实施的应用之间, ...

  3. 《TensorFlow2深度学习》学习笔记(四)对笔记二中的模型增加正确率展示

    全部代码如下:(红色部分为与笔记二不同之处) #1.Import the neccessary libraries needed import numpy as np import tensorflo ...

  4. SMBus PEC

    SMBus一种I2C总线的变种 SMBus 提供了PEC方式,提高了传输的可靠性. 总线的发展都是在提高速度,提高可靠性或者提高传输效率上下功夫. PEC不具备纠错的能力,是在I2C link lay ...

  5. Django设置应用名与模型名为中文

    修改polls包里面的apps.py: from django.apps import AppConfig class PollsConfig(AppConfig): name = 'polls' v ...

  6. MySQL数据的优化方案

    一.选取最使用的字段属性 mysql可以使用的支持大数据量的存取,但是一般说来,数据库中的表越小,在它上面执行的查询也就会越快,因此在创建表的时候,为了获得更好的性能,我们可以将表中的字段的宽度尽量设 ...

  7. Json在序列化注意问题

    Java中的Json序列化,不容忽视的getter 问题重现 public class AjaxJson { private boolean success; private String msg; ...

  8. Dynamics CRM 数据数量限制更改

    1.在CRM2016中如果想要导出超过10000记录数据,更新 MaxRecordsForExportToExcel  这个字段的值. SELECT MaxRecordsForExportToExce ...

  9. jedis的连接池

    1.需要先打开虚拟机,并开启Linux系统的端口号:6379: 其中,第一行代码为修改字符编码格式,解决SSH中文乱码问题. 2.开启redis: 3.利用连接池实现数据的存取: (1)代码实现: i ...

  10. Js 在页面中输入消息的几种方式

    一.方式 alert(“”); confirm(“”) ; prompt(“”);         接收用户信息 console.log(“”);      在网页控制台中输出消息 document. ...