排序算法7--选择排序--堆排序

堆排序

1.堆排序是一种树形选择排序,在排序过程中,将待排序的记录r[1..n]看成是一棵完全二叉树的顺序存储结构,利用完全二叉树的双亲结点和孩子结点之间的内在关系,在当前无序的序列中选择关键字最大(或者最小的)记录

   1.1、先将无序队列,列成二叉树的的样子
  1.2、建初堆。从n/2向上取整处开始调整,当前节点数字一定要大于(小于)左右两孩子,n/2--,一直向上若最后排序结果是升序就调整成大顶堆,若是降序就调整成小顶堆
  1.3、开始排序,将堆顶元素和最后一个元素互换位置,换下来的即可以依次加入有序队列,交换完成后继续进行第二步操作,调整堆
  1.4、然后再进行交换(注意:上面交换下来的已经是排好序的,不用进行调整,即忽略交换下来的数字即可)

由于堆排序的过程图画着不太方便,就上传了数据结构课本上的堆排序过程,这个比较过程比较详细

建初堆:

堆排序调整堆

2.时间复杂度

   堆排序的时间主要耗费在建初堆和调整堆时进行的反复“筛选”上
  设有n个记录的初始序列所对应的完全二叉树的深度为h,建处堆时,每个非终端节点都要自上而下进行“筛选”,由于第i层上的节点小于等于2的i-1次方,。。。。。。
  堆排序最坏的情况下时间复杂度也是O(nlog2n)
  空间复杂度O(1)

具体时间复杂度等分析,请参考:http://www.cnblogs.com/zhangxue521/p/6748085.html

3.算法特点

  ①具有不稳定性
  ②只能用于顺序结构,不能用于链式结构
  ③建初堆时所需的比较次数比较多,因此记录数较少时不宜采用。堆排序在最坏的情况下的时间复杂度为O(nlog2n),因此当记录较多时比较高效

4.java代码

 1 package 平时常用;
 2
 3 public class _4堆排序 {
 4    public static void main(String[] args) {
 5             _4堆排序 hs = new _4堆排序();
 6             int[] array = {87,45,78,32,17,65,53,9,122};
 7             System.out.print("构建大根堆:");
 8             hs.toString(hs.buildMaxHeap(array));
 9             System.out.print("\n"+"删除堆顶元素:");
10             hs.toString(hs.deleteMax(array));
11             System.out.print("\n"+"插入元素63:");
12             hs.toString(hs.insertData(array, 63));
13             System.out.print("\n"+"大根堆排序:");
14             hs.toString(hs.heapSort(array));
15    }
16    //输出
17    public void toString(int[] array){
18        for(int i:array){
19            System.out.print(i+" ");
20        }
21    }
22  //构建大根堆:将array看成完全二叉树的顺序存储结构
23    private int[] buildMaxHeap(int[] array){
24        //即从中间元素开始调整
25        //从最后一个节点array.length-1的父节点(array.length-1-1)/2开始,直到根节点0,反复调整堆
26        for(int i=(array.length-2)/2;i>=0;i--){
27            adjustDownToUp(array, i,array.length);
28        }
29        return array;
30    }
31
32    //将元素array[k]自下往上逐步调整树形结构
33    private void adjustDownToUp(int[] array,int k,int length){
34        int temp = array[k];
35        for(int i=2*k+1; i<length-1; i=2*i+1){    //i为初始化为节点k的左孩子,沿节点较大的子节点向下调整
36            if(i<length && array[i]<array[i+1]){  //取节点较大的子节点的下标
37                i++;   //如果节点的右孩子>左孩子,则取右孩子节点的下标
38            }
39            if(temp>=array[i]){  //根节点 >=左右子女中关键字较大者,调整结束
40                break;
41            }else{   //根节点 <左右子女中关键字较大者
42                array[k] = array[i];  //将左右子结点中较大值array[i]调整到双亲节点上
43                k = i; //【关键】修改k值,以便继续向下调整
44            }
45        }
46        array[k] = temp;  //被调整的结点的值放人最终位置
47    }
48
49  //堆排序
50    public int[] heapSort(int[] array){
51        array = buildMaxHeap(array); //初始建堆,array[0]为第一趟值最大的元素
52        for(int i=array.length-1;i>1;i--){
53            int temp = array[0];  //将堆顶元素和堆低元素交换,即得到当前最大元素正确的排序位置
54            array[0] = array[i];
55            array[i] = temp;
56            adjustDownToUp(array, 0,i);  //整理,将剩余的元素整理成堆
57        }
58        return array;
59    }
60
61  //删除堆顶元素操作
62    public int[] deleteMax(int[] array){
63        //将堆的最后一个元素与堆顶元素交换,堆底元素值设为-99999
64        array[0] = array[array.length-1];
65        array[array.length-1] = -99999;
66        //对此时的根节点进行向下调整
67        adjustDownToUp(array, 0, array.length);
68        return array;
69    }
70
71  //插入操作:向大根堆array中插入数据data
72    public int[] insertData(int[] array, int data){
73        array[array.length-1] = data; //将新节点放在堆的末端
74        int k = array.length-1;  //需要调整的节点
75        int parent = (k-1)/2;    //双亲节点
76        while(parent >=0 && data>array[parent]){
77            array[k] = array[parent];  //双亲节点下调
78            k = parent;
79            if(parent != 0){
80                parent = (parent-1)/2;  //继续向上比较
81            }else{  //根节点已调整完毕,跳出循环
82                break;
83            }
84        }
85        array[k] = data;  //将插入的结点放到正确的位置
86        return array;
87    }
88
89
90
91 }
时间: 2024-11-01 15:23:15

排序算法7--选择排序--堆排序的相关文章

数据结构排序算法之选择排序

今天继续介绍一种排序算法:选择排序. 选择排序的基本思想就是从待排序列中选择出最小的,然后将被选出元素和序列的第一个元素互换位置(当前默认是升序排列),则互换完成后第一个元素就是整个序列的最小的元素,则一次选择排序结束.然后我们从剩下的子序列中选择出最小的,然后将该被选出来的元素和该子序列的第一个元素(即整个序列的第二个元素)互换位置,则当前整个序列的第二个元素就是当前序列中的次最小值,第二次选择排序结束.以此类推,直到该待排序列只剩下一个元素后,则整个序列有序. 具体过程如下图所示: 下面就不

排序算法之选择排序

一. 算法描述 选择排序:在一个长度为N的无序数组中,在第一趟遍历N个数据,找出其中最小的数值与第一个元素交换,第二趟遍历剩下的N-1个数据,找出其中最小的数值与第二个元素交换......第N-1趟遍历剩下的2个数据,找出其中最小的数值与第N-1个元素交换,至此选择排序完成. 二. 算法分析 平均时间复杂度:O(n2) 空间复杂度:O(1)  (用于交换和记录索引) 稳定性:不稳定 (比如序列[5, 5, 3]第一趟就将第一个[5]与[3]交换,导致第一个5挪动到第二个5后面) 三. 算法实现

【排序算法】选择排序(Selection sort)

0. 说明 选择排序(Selection sort)是一种简单直观的排序算法. 它的工作原理如下. 首先在未排序序列中找到最小(大)元素,存放到排序序列的起始位置,然后,再从剩余未排序元素中继续寻找最小(大)元素,然后放到已排序序列的末尾.以此类推,直到所有元素均排序完毕. 选择排序的主要优点与数据移动有关.如果某个元素位于正确的最终位置上,则它不会被移动.选择排序每次交换一对元素,它们当中至少有一个将被移到其最终位置上,因此对 n 个元素的表进行排序总共进行至多 n-1 次交换.在所有的完全依

初级排序算法之选择排序

初级排序算法 本质是对要排序的数组进行嵌套循环,内层循环负责局部的排序,外层循环负责剩余的无序元素的递减.所以你只要理解嵌套循环和比较大小就能很快的掌握初级排序算法. 选择排序 一个无序的数组 a = [0, 4, 6, 3, 8, 2, 3, 9], 你也可以把a的元素想象成任何现实中可比较的具体物体.例如,有10根长短不一的木条,我们如何对它们进行排序?一个最直接的思想,先拿出最短的放到最前面,在剩余的木条中再拿出最短的放在第二位...直到最后一根木条.从中我们可以看出,1. 我们需要再一次

【排序算法】选择排序

选择排序算法原理 选择排序算法时间复杂度分析 选择排序算法稳定性分析 选择排序算法C语言代码 #include <stdio.h> //交换两个元素的值 void swap(int* a, int* b) { int temp; temp = *a; *a = *b; *b = temp; } void selectionSort(int arr[], int length) { int i, j, maxIndex; for(i = length; i > 0; i--) { //假设

选择排序算法---直接选择排序和堆排序

本文主要是解析选择排序算法:直接选择排序和堆排序. 一.直接选择排序   基本思想:       选择排序(Selection sort)是一种简单直观的排序算法.它的工作原理如下.首先在未排序序列中找到最小(大)元素,存放到排序序列的起始位置,然后,再从剩余未排序元素中继续寻找最小(大)元素,然后放到已排序序列的末尾.以此类推,直到所有元素均排序完毕. 选择排序的主要优点与数据移动有关.如果某个元素位于正确的最终位置上,则它不会被移动.选择排序每次交换一对元素,它们当中至少有一个将被移到其最终

排序算法——直接选择排序

直接选择排序每一趟排序都会从未排序的序列中选择出最小的元素来,然后跟未排序序列的第一个元素交换.这样经过n-1趟排序后,每趟排序选择出的 最小元素便成了有序的序列. 算法实现如下: #include <stdio.h> #include <stdlib.h> void SelectSort(int A[],int n) { int i, j, index, temp; for(i = 0; i < n-1; i++) // 进行n-1趟排序 { index = i; // 辅

经典排序算法--简单选择排序

算法简介 简单选择排序是一种选择排序. 选择排序:每趟从待排序的记录中选出关键字最小的记录,顺序放在已排序的记录序列末尾,直到全部排序结束为止. 白话理解 依然已排队为例,在排队时,有的老师可能会选择这样排序,先在一列中选出最矮的,放在第一位,然后选出第二矮的,放在第二位.队伍完成排序.而这就是选择排序的思想. 简单排序处理流程 (1)从待排序序列中,找到关键字最小的元素: (2)如果最小元素不是待排序序列的第一个元素,将其和第一个元素互换: (3)从余下的 N - 1 个元素中,找出关键字最小

算法学习之排序算法:选择排序

选择排序:每一趟在n-i+1(i=1,2,...,n-1)个记录中选取关键字最小的记录作为有序序列中第i个记录. 一.简单选择排序 一趟选择排序操作: 通过n-i次关键字间的比较,从n-i+1个记录中选出关键字最小的记录,并和第i(1<=i<=n)个记录交换之. 对L[1...n]中记录进行简单选择排序的算法为:令i从1至n-1,进行n-1趟选择操作.简单选择排序过程中,所需进行记录移动的操作次数较少,然而,无论记录的初始排列如何,所需关键字间的比较次数相同.因此,总的时间复杂度为O(n^2)

排序算法总结------选择排序 ---javascript描述

每当面试时避不可少谈论的话题是排序算法,上次面试时被问到写排序算法,然后脑袋一懵不会写,狠狠的被面试官鄙视了一番,问我是不是第一次参加面试,怎么可以连排序算法都不会呢?不过当时确实是第一次去面试,以此恶补排序算法. 一.基本排序算法:基本排序算法有冒泡排序,选择排序,插入排序. 选择排序算法思想:选择排序从数组的头开始,将第一个元素与其他元素进行比较,找到最小的元素,然后放到数组的第一个位置.然后再从第二个元素开始找比他小的元素,放到第二个位置,以此类推,重复上述步骤,当进行到数组的第二个位置时