/**
  *堆排序思路:O(nlogn)
  *    用最大堆,传入一个数组,先用数组建堆,维护堆的性质
  *    再把第一个数与堆最后一个数调换,因为第一个数是最大的
  *    把堆的大小减小一
  *    再  在堆的大小上维护堆的性质
  *    重复操作..
  *
  *
  */
 public  class  HeapSort
 {
     /**
      *静态变量存放堆的大小
      */
     private  static  int  heapsize = 0 ;    

     /**
      *堆排序主方法
      *    构建最大堆,然后进行堆排序
      *    堆排序是把最上面的最大的元素放在最下面,然后再维护上面最大堆的性质
      */
     public  static  void  heapSort(int[] resouceArr)
     {
         //堆的大小
         heapsize = resouceArr.length - 1 ;

         _buildMaxHeap(resouceArr);

         for( int i = resouceArr.length - 1 ; i >= 0 ; i--)
         {
             int temp = resouceArr[0] ;
             resouceArr[0] = resouceArr[i] ;
             resouceArr[i] = temp ; 

             heapsize = heapsize - 1 ;
             _maxHeapify( resouceArr, 0 );
         }
     }

     /**
      *构建最大堆
      *    构建之后还不是有序的,但符合最大堆性质,上面的数一定大于下面的数
      */
     private  static  void  _buildMaxHeap(int[] arr)
     {
         int length = arr.length - 1 ; 

         //从倒数第二排开始维护最大堆性质
         //    当heapsize为偶数时,heapsize要减一
         //    当heapsize为奇数时,不变
         if(length % 2 == 0)
         {
             length--;
         }
         for( int i = length / 2 ; i >= 0  ; i--)
         {
             _maxHeapify(arr , i );
         }
     }

     /**
      *用于维护堆的性质
      *    树形堆在postion的位置向下维护堆的性质,自postion向下满足最大堆的性质
      */
     private  static  void  _maxHeapify(int[] arr,int postion)
     {
         //计算postion的左孩子和右孩子
         postion = postion + 1 ;

         int  l = postion * 2 - 1;
         int  r = postion * 2 ;
         postion = postion - 1 ;

         int largest = maxNumInThreeNum(arr , postion , l , r);

         //如果不满足最大堆性质,则交换值,然后检查树形下方是否满足最大堆性质
         if(largest <= heapsize)
         {
             if( largest != postion)
             {
                 //交换最大值与父节点值
                 int  temp = arr[postion] ;
                 arr[postion] = arr[largest] ;
                 arr[largest] = temp ;
                 //如果子节点变动了,则重新构建已子节点为根的最大堆
                 _maxHeapify(arr , largest);                

             }
         }
     }    

     /**
      *比较数组中的三个数找出最大值
      */
     private  static  int  maxNumInThreeNum(int[] arr ,int a, int b , int c)
     {
         int max = a ;
         //数组长度小于左孩子,最大值为本身
         if(heapsize < b)
         {
             max = a ;
         }
         else if(heapsize >=b && heapsize < c)
         {
             if(arr[a] < arr[b])
             {
                 max = b ;
             }
         }
         else
         {
             if(arr[a] > arr[b])
             {
                 max = a ;
             }
             else
             {
                 max = b ;
             }
             if(arr[max] < arr[c])
             {
                 max = c ;
             }
         }
         return max ;
     }
 }

排序算法FOUR:堆排序HeapSort的更多相关文章

  1. Java常见排序算法之堆排序

    在学习算法的过程中,我们难免会接触很多和排序相关的算法.总而言之,对于任何编程人员来说,基本的排序算法是必须要掌握的. 从今天开始,我们将要进行基本的排序算法的讲解.Are you ready?Let ...

  2. 排序算法之堆排序(Heapsort)解析

    一.堆排序的优缺点(pros and cons) (还是简单的说说这个,毕竟没有必要浪费时间去理解一个糟糕的的算法) 优点: 堆排序的效率与快排.归并相同,都达到了基于比较的排序算法效率的峰值(时间复 ...

  3. Java排序算法之堆排序

    堆的概念: 堆是一种完全二叉树,非叶子结点 i 要满足key[i]>key[i+1]&&key[i]>key[i+2](最大堆) 或者 key[i]<key[i+1] ...

  4. 《排序算法》——堆排序(大顶堆,小顶堆,Java)

    十大算法之堆排序: 堆的定义例如以下: n个元素的序列{k0,k1,...,ki,-,k(n-1)}当且仅当满足下关系时,称之为堆. " ki<=k2i,ki<=k2i+1;或k ...

  5. C++编程练习(13)----“排序算法 之 堆排序“

    堆排序 堆是具有下列性质的完全二叉树:每个结点的值都大于或等于其左右孩子结点的值,称为大顶堆(也叫最大堆):或者每个结点的值都小于或等于其左右孩子结点的值,称为小顶堆(也叫最小堆). 最小堆和最大堆如 ...

  6. 数据结构与算法之PHP排序算法(堆排序)

    一.堆的定义 堆通常是一个可以被看做一棵树的数组对象,其任一非叶节点满足以下性质: 1)堆中某个节点的值总是不大于或不小于其父节点的值: 每个节点的值都大于或等于其左右子节点的值,称为大顶堆.即:ar ...

  7. 八大排序算法之七—堆排序(Heap Sort)

    堆排序是一种树形选择排序,是对直接选择排序的有效改进. 基本思想: 堆的定义如下:具有n个元素的序列(k1,k2,...,kn),当且仅当满足 时称之为堆.由堆的定义可以看出,堆顶元素(即第一个元素) ...

  8. 排序算法(2) 堆排序 C++实现

    堆 1 数组对象 2 可以视为一棵完全二叉树 3 一个堆可以被看作一棵二叉树和一个数组,如下图所示: 4 下标计算(通常使用内联函数或者宏来定义下标操作): 已知某个结点的下标为i 其父节点下标:i/ ...

  9. Python 一网打尽<排序算法>之堆排序算法中的树

    本文从树数据结构说到二叉堆数据结构,再使用二叉堆的有序性对无序数列排序. 1. 树 树是最基本的数据结构,可以用树映射现实世界中一对多的群体关系.如公司的组织结构.网页中标签之间的关系.操作系统中文件 ...

随机推荐

  1. 【转】C++及java在内存分配上的区别

    转自:http://blog.csdn.net/qinghezhen/article/details/9116053 C++内存分配由五个部分组成:栈.堆.全局代码区.常量区.程序代码区.如下图所示: ...

  2. Android 如何添加一种锁屏方式

    前言          欢迎大家我分享和推荐好用的代码段~~ 声明          欢迎转载,但请保留文章原始出处:          CSDN:http://www.csdn.net        ...

  3. js获取光标位置例子

    <html><head><title>TEST</title><style>body,td { font-family: verdana, ...

  4. Ubuntu10.04下载并编译Android4.3源代码

    注:转载或引用请标明出处    http://blog.csdn.net/luzhenrong45/article/details/9719433 去年用Ubuntu10.10成功下载并编译Andro ...

  5. android平台获取手机IMSI,IMEI ,序列号,和 手机号的方法

    1)获取运营商sim卡imsi号, String IMSI =android.os.SystemProperties.get( android.telephony.TelephonyPropertie ...

  6. SerialPort基本小例

    SerialPort是用于串口通信的控件与VB6中的MSCOMM控件相似,使用很方便... vb.net CodeImports System.IO.PortsImports System.TextP ...

  7. ASP.NET 不同页面之间传值

    不同页面之间如何传值?我们假设A和B两个页面,A是传递,B是接收. 下面学习4种方式: 通过URL链接地址传递 POST方式传递 Session方式传递 Application方式传递 1. 通过UR ...

  8. hadoop学习记录(零)

    这个博客开通快一年了,但是由于种种原因,始终没有能够养成定期更新的习惯. 最近完成了比赛的项目,向除了android开发以外再拓宽一下自己的技能树. 最近购买了<java8函数式编程>和& ...

  9. 【安卓小技巧】WebView设置在本页面打开网页,而不是启动浏览器打开

    使用WebView可以巧妙的在安卓APP中嵌入HTML页面, WebView wb = (WebView) findViewById(R.id.web); //找到WebView控件 wb.setWe ...

  10. PHP之ThinkPHP数据操作CURD

    两个数据表 具体操作如下: /**********************数据库操作********************/       ////////添加数据//////////         ...