排序算法---(四)

引言

在前几篇文章里面讲到了六种排序,今天来讲一下剩下两种:基数排序、堆排序

基数排序

1.思路

(1)首先确定最大数的位数:找到待排序数组中的最大数,并确定其位数

(2)将元素按照相应的位数存入桶中:先定义一个桶(二维数组):用来存放待排序数组元素,再定义一个桶计数器(一维数组):用来记录每个桶里面有几个元素,将元素放入桶中有三步:① 放哪个桶(看相应位数是多少,例如相应位数上是1,那么就放在x坐标为1的桶里)、② 找到桶应该放在桶内的哪个位置(因为相应位数上是同一个数字的可能有多个,因此桶计数器发挥作用了,若计数为3,那说明这个桶已经放了三个元素了,那将待放入元素放在第四个位置上)、③ 更新桶计数器(+1);

(3)相应位数排完序后,将存入桶里元素有序的放回原数组

2.代码

java 复制代码
public class jsSort {
    public static void main(String[] args) {
        int[] arr=new int[]{5,1,3,9,8,2};
        int n=arr.length;
        sort(arr);
        for(int i=0;i<n;i++){
            System.out.print(arr[i]+" ");
        }
    }

    public static void sort(int[] arr) {
        int max=arr[0];
        for(int i=0;i<arr.length;i++){
            if(arr[i]>max){
                max=arr[i];
            }
        }

        int maxLength=(max+"").length();
        int[][] bucket=new int[10][arr.length];
        int[] bucketCount=new int[10];
        int t=1;
        for(int i=0;i<maxLength;i++){
            //存数据
            for(int j=0;j<arr.length;j++){
                int element=arr[j]/t%10;
                bucket[element][bucketCount[element]]=arr[j];
                bucketCount[element]++;
            }
            int index=0;
            //取数据
            for(int k=0;k<10;k++){
                if(bucketCount[k]!=0){
                    for(int j=0;j<bucketCount[k];j++){
                        arr[index++]=bucket[k][j];
                    }
                }
                bucketCount[k]=0;
            }
            t=t*10;
        }
    }
}

堆排序

1.相关知识

完全二叉树:从上到下、从左到右依次填满结点,前面每一层必须是满的,最后一层可以不满,但只能缺右边的结点,不能中间空、不能左边空

大顶堆:在完全二叉树的基础上父节点的值大于左右孩子的值

2.思路

(1)首先将待排序数组构造成一个大顶堆,此时整个数组最大值就被放在了二叉树的顶端

(2)将堆顶元素和堆底元素进行交换,此时堆底元素变成了整个数组最大值

(3)重新构建大顶堆,不包含上一步的堆底元素(最大值),重复上述步骤,这样就实现了每次都把最大元素拿出来,实现了数组升序

3.关键点

构造大顶堆:从最后一个元素当作parent,首先判断是否有孩子:child=parent*2+1是否小于nums.length,直到找到符合条件的parent,找到后开始判断parent和child哪个大,大的换到parent位置,依次遍历,遍历到二叉树的最顶端(数组的第一个元素),这样不断地向上遍历并交换,就实现了堆顶元素是数组最大的元素

排除上一轮的堆底元素:只需要将待排序数组长度减一即可

4.代码

java 复制代码
public class heapSort {
    public static void main(String[] args) {
        int[] arr=new int[]{5,1,3,9,6,8,2};

        for(int i=arr.length-1;i>=0;i--){
            sort(arr,i,arr.length);
        }

        //交换堆顶元素和堆底元素,并排除堆底元素
        for(int k=arr.length-1;k>=0;k--){
            int temp=arr[k];
            arr[k]=arr[0];
            arr[0]=temp;
            sort(arr,0,k);
        }
        //循环输出
        for(int j=0;j<arr.length;j++){
            System.out.print(arr[j]+" ");
        }
    }
    public static void sort(int[] arr,int parent,int length){
        int child=parent*2+1;

        while(child<length){
            int rChild=child+1;
            if(rChild<length && arr[rChild]>arr[child]){
                child++;
            }
            
            //将大的元素放在parent位置
            if(arr[child]>arr[parent]){
                int temp=arr[child];
                arr[child]=arr[parent];
                arr[parent]=temp;

                parent=child;
                child=parent*2+1;
            }else{
                break;
            }
        }
    }
}

小舟有话说

至此,八大排序已全部讲解完毕,其中要求我们熟练掌握的有三种排序:冒泡排序、快速排序、堆排序,大家一定要熟练记忆,堆排序虽然看起来难理解,但是只要懂了二叉树相关知识就不难。

如果觉得内容不错,那就点点赞,点点关注吧,下次找我不迷路~

相关推荐
童话ing2 小时前
【LeetCode】239.滑动窗口最大值
数据结构·算法·leetcode·golang
计算机安禾2 小时前
【数据结构与算法】第13篇:栈(三):中缀表达式转后缀表达式及计算
c语言·开发语言·数据结构·c++·算法·链表
章鱼丸-2 小时前
DAY40 训练与测试规范写法
人工智能·算法·机器学习
代码飞天2 小时前
算法与数据结构之又臭又长的表
数据结构·算法
A923A2 小时前
【洛谷刷题 | 第七天】
算法·模拟·洛谷
故事和你912 小时前
洛谷-入门4-数组3
开发语言·数据结构·c++·算法·动态规划·图论
玉树临风ives2 小时前
atcoder ABC 451 题解
c++·算法·atcoder
_日拱一卒2 小时前
LeetCode:和为K的子数组
算法·leetcode·职场和发展
周可温8413 小时前
动手学RAG
算法