zoukankan      html  css  js  c++  java
  • Java实现堆排序和计数排序

    堆排序代码:

      思想:每次都取堆顶的元素,将其放在序列最后面,然后将剩余的元素重新调整为最小堆,依次类推,最终得到排序的序列。

    import java.util.Arrays;
    
    /**
     * 思路:首先要知道大顶堆和小顶堆,数组就是一个堆,每个i节点的左右孩子是2i+1和2i+2
     *         有了堆,将其堆化:从(n/2)-1个元素开始向下修复,将每个节点修复为小(大)顶堆
     *         修复完成后,数组具有小(大)顶堆的性质
     *         按序输出:小顶堆可以对数组逆序排序,每次交换堆顶和末尾元素,对堆顶进行向下修复,这样次小元素又到堆顶了
     * 
     * 时间复杂度:堆化:一半的元素修复,修复是单分支的,所以整体堆化为nlgn/2
     * 排序:n个元素都要取出,因此调整n次,每次调整修复同上是lgn的,整体为nlgn
     * 空间复杂度:不需要开辟辅助空间
     * 原址排序
     * 稳定性
     *
     */
    public class HeapSort {
    
        static void sort(int []A){
            // 堆排序第一步: 先对A进行堆化
            makeMinHeap(A);
            for(int x = A.length-1;x>=0;x--){
                // 堆排序第二步: 把堆顶,0号元素和最后一个元素对调
                swap(A, 0, x);
                // 堆排序第三步:缩小堆的范围,对堆顶元素进行向下调整
                MinHeapFixDown(A, 0, x);
            }
        }
        
        static void makeMinHeap(int[] A){
            int n = A.length;
            for(int i = n/2-1;i>=0;i--){
                MinHeapFixDown(A,i,n);
            }
        }
        
        private static void MinHeapFixDown(int[] A, int i, int n) {
            //  找到左右孩子
            int left = 2 * i + 1;
            int right = 2 * i + 2 ;
            // 左孩子已经越界,i就是叶子节点
            if (left>=n) {
                return ;
            }
            // min 指向了左右孩子中较小的那个
            int min = left;
            if (right>=n) {
                min = left;
            }else {
                if (A[right]<A[left]) {
                    min = right;
                }
            }
            // 如果A[i]比两个孩子都要小,不用调整
            if (A[i]<=A[min]) {
                return ;
            }
            // 否则,找到两个孩子中较小的,和i交换
            int temp = A[i];
            A[i] = A[min];
            A[min] = temp;
            // 小孩子那个位置的值发生了变化,i变更为小孩子那个位置,递归调整
            MinHeapFixDown(A, min, n);
        }
        
        
        
        private static void swap(int[] A, int p, int bigger) {
            int temp = A[p];
            A[p] = A[bigger];
            A[bigger] = temp;
            
        }
    
        public static void main(String[] args) {
            int arr[] = new int[10];
            for(int i=0;i<10;i++){
                arr[i] = (int) ((Math.random()+1)*10);
            }
            
            
            System.out.println("排序前:"+Arrays.toString(arr));
            sort(arr);
            System.out.println("排序后:"+Arrays.toString(arr));
        }
    
    }

    堆排序结果:

      

    计数排序代码:

    import java.util.Arrays;
    
    /**
     * 计数排序
     * 思路:开辟新的空间,空间大小为max(source)+1
     *         扫描source,将value作为辅助空间的下标,用辅助空间的该位置元素记录value的个数
     *         如 9 7 5 3 1,helper的空间就为10
     *         依次扫描,value为9,将helper[9]++,以此类推,完成之后,再去遍历helper
     *         如果该位(index)的值为0,说明index不曾在source中出现
     *         如果该位(index)的值为 1,说明出现了1次,为2说明出现了两次
     * 时间复杂度:扫描一次source,扫描一次helper,复杂度为N+K
     * 空间复杂度:如果source里面有个元素较大的,那么开辟的辅助空间较大
     * 非原址排序
     * 稳定性:相同元素不会出现交叉,非原址都是拷来拷去
     * 如果要优化一下空间,可以求出minOf(source),那么helper的长度为(max-min)+1,这样就能短点
     * 计数有缺陷,数据较为密集或范围较小时,适用。
     */
    public class CountSort {
    
        static void sort(int []source){
            int max = source[0];
            for (int i = 1; i < source.length; i++) {
                if (source[i]>max) {
                    max = source[i];
                }
            }
            int []helper = new int[max+1];
            for(int e:source){
                helper[e]++;
            }
            int current = 0;  // 数据回填的位置
            for (int i = 1; i < helper.length; i++) {
                while(helper[i]>0){
                    source[current++] = i;
                    helper[i]--;
                }
            }
        }
        
        // 保证排序稳定性的版本
        public static void sort2(int[] source) {
            int max = source[0];
            for (int i = 1; i < source.length; i++) {
                if (source[i]>max) {
                    max = source[i];
                }
            }
            int []helper = new int[max+1];
            for (int e : source) {
              helper[e]++;
            }
            for (int i = 1; i < helper.length; i++) {
              helper[i] += helper[i - 1];
            }
            int len = source.length;
            int[] target = new int[len];
            for (int i = len - 1; i >= 0; i--) {
              target[helper[source[i]] - 1] = source[i];
              helper[source[i]]--;
            }
            System.arraycopy(target, 0, source, 0, len);
          }
        
        public static void main(String[] args) {
            int arr[] = new int[10];
            for(int i=0;i<10;i++){
                arr[i] = (int) ((Math.random()+1)*10);
            }
            System.out.println("排序前:"+Arrays.toString(arr));
            sort2(arr);
            System.out.println("排序后:"+Arrays.toString(arr));
        }
    
    }

    计数排序结果:  

      

  • 相关阅读:
    Oracle数据库对表字段的操作命令
    解决eclipse中git中cannot open gituploadpack(无法打开Git上传包)问题
    C# 启动外部程序的几种方法
    php面试题及答案(转)
    每日知识(1)半结构化数据
    每日知识(2)云计算
    方法对象
    什么是MA移动平均线它的特点及应用
    My first testcase about C#&C++
    局部变量的角色
  • 原文地址:https://www.cnblogs.com/xiaoyh/p/10281647.html
Copyright © 2011-2022 走看看