【Java数据结构】想进大厂必须牢记于心的——常见八大排序算法(上)

简介: 笔记

🛸基本概念


⭐排序


排序,就是使一串记录,按照其中的某个或某些关键字的大小,递增或递减的排列起来的操作。

平时的上下文中,如果提到排序,通常指的是排升序(非降序)。

通常意义上的排序,都是指的原地排序(in place sort)。


⭐稳定性


两个相等的数据,如果经过排序后,排序算法能 保证其相对位置不发生变化 ,则我们称该算法是具备 稳定性 的排序算法20.png


🛸七大基于比较的排序

21.png


⭐插入排序


🎄1. 直接插入排序


思路:


插入排序基本思想是将一个记录插入到已经排好序的有序表中,从而变成一个新的、记录数增1的有序表。

在其实现过程使用双层循环,外层循环对 除了第一个元素之外的所有元素,内层循环对 当前元素前面有序表进行待插入位置查找,并进行移动

图解:

22.gif


代码实现:


22.gif

     /**
     * 时间复杂度:
     *        最好:O(N) -> 数据是有序的
     *
     *        最坏:O(N^2) -> 无序的数据
     *
     * 空间复杂度:O(1)
     * 稳定性:稳定排序
     * @param array
     */
    //插入排序
    public static void insertSort (int[]array){
        for (int i = 1; i<array.length; i++){//外循环
        //从1开始表示:假设array[0] 已经放好位置了
        //后面的数字就是插入到它左边还是右边的问题。
            int tmp = array[i];//设置一个缓存tmp
            int j = i-1;
            for (; j >=0 ; j--){//内循环
                if (array[j]>tmp){//如果array[j]大于缓存值,说明要换位置
                    array[j+1] = array[j];
                }else{//否则直接退出当前这一次的循环
                    break;
                }
            }
            //最后记得要把缓存值插入到表中
            array[j+1] = tmp;//j此时有可能已经是-1了,所以要变成0下标就得+1
        }
    }

🎄2. 希尔排序(直接插入排序的优化)


思路:

希尔排序法又称缩小增量法。

希尔排序法的基本思想是:

先选定一个整数(gap),把待排序文件中所有记录分成gap个组,所有距离为gap的记录分在同一组内,并对每一组内的记录进行排序。然后,取gap / 2,重复上述分组和排序的工作。当gap到达1时,所有记录在同一组内排好序。

注意gap的问题:

23.png

1.希尔排序是对直接插入排序的优化。

2.当gap > 1时都是预排序,目的是让数组更接近于有序。当gap == 1时,数组已经接近有序的了,这时 相当于直接用插入排序,这样就会很快,因为 直接插入排序是越有序越快。这样整体而言,可以达到优化的效果。我们实现后可以进行性能测试的对比。

图解:

24.png

代码实现:

/**
     * @param array 排序的数组
     * @param gap   每组的间隔  -》 组数
     */
    public static void shell(int[] array,int gap) {
    //如果将gap全部换成1,会发现其实就是直接插入排序
    //所以当gap到1的时候,这就表示这是最后一次排序
    //这最后一次排序其实就是一个直接插入排序
        for (int i = gap; i < array.length; i++) {
            int tmp = array[i];
            int j = i-gap;
            for (; j >= 0; j -= gap) {
                if(array[j] > tmp) {
                    array[j+gap] = array[j];
                }else {
                    break;
                }
            }
            array[j+gap] = tmp;
        }
    }
    /**
     * 时间复杂度:不好算  n^1.3 - n^1.5 之间
     * 空间复杂度:O(1)
     * 稳定性:不稳定的排序
     *      技巧:如果在比较的过程当中 没有发生跳跃式的交换 那么就是稳定的
     * @param array
     */
    public static void shellSort(int[] array) {
        //处理gap
        int gap = array.length;
        while (gap > 1) {
            gap /= 2;//保证最后一个序列间隔是 1  除几都行
            shell(array,gap);
        }
    }


⭐选择排序


🎄3. 选择排序


思路:

将一组数据分为有序区(排过序的数据)和无序区(未排序数据),每一次从无序区间选出最大(或最小)的一个元素,存放在无序区间的最后(或最前),直到全部待排序的数据元素排完 。


选择排序的步骤:

1>首先在未排序序列中找到最小(大)元素,存放到排序序列的起始位置。

2>再从剩余未排序元素中继续寻找最小(大)元素,然后放到未排序序列的起始位置。

3>重复第二步,直到所有元素均排序完毕。


图解:

25.gif

代码实现:

/**
     * 时间复杂度:
     *      最好:O(N^2)
     *      最坏:O(N^2)
     * 空间复杂度:O(1)
     * 稳定性:不稳定的
     * @param array
     */
    public static void selectSort(int[] array) {
        for (int i = 0; i < array.length; i++) {//下标i前边的为有序区间
            for (int j = i+1; j < array.length; j++) {
                if(array[j] < array[i]) {
                    int tmp = array[i];
                    array[i] = array[j];
                    array[j] = tmp;
                }
            }
        }
    }


🎄4.堆排序(如果不了解堆的话可以看看我上一篇讲 堆 的博客)


思路:


准备知识:

堆的结构可以分为大根堆和小根堆,是一个 完全二叉树,而堆排序是根据堆的这种数据结构设计的一种排序,下面先来看看什么是大根堆和小根堆


性质:

每个结点的值都大于其左孩子和右孩子结点的值,称之为大根堆;

每个结点的值都小于其左孩子和右孩子结点的值,称之为小根堆。

如下图

50.png

我们对上面的图中每个数都进行了标记,上面的结构映射成数组就变成了下面这个样子

51.png

基本步骤:


首先将待排序的数组构造成一个大根堆(升序用大根堆,降序就用小根堆),此时,整个数组的最大值就是堆结构的顶端

将顶端的数与末尾的数交换,此时,末尾的数为最大值,将末尾这个最大值提取出去,剩余待排序数组个数为n-1

将剩余的n-1个数再构造成大根堆,再将顶端数与n-1位置的数交换,如此反复执行,便能得到有序数组

图解:52.gif

代码实现:

    //堆的向下调整
    public static void siftDown(int[] array,int root,int len) {//len表示末尾元素下标
        int parent = root;
        int child = 2*parent+1;//先找到左孩子节点
        while (child <= len) {//当child>length的时候说明当前子树已经调整好了
             //先根据左孩子节点判断右孩子节点是否存在,且是否大于左孩子节点
            if(child+1 <= len && array[child] < array[child+1]) {//如果存在,且值大于左孩子节点
                child++;
            }
            //child的下标就是左右孩子的最大值下标
            if(array[child] > array[parent]) {//如果孩子节点最大值,大于父节点,则要交换位置,因为要建大根堆
                int tmp = array[child];
                array[child] = array[parent];
                array[parent] = tmp;
                //继续向下看是否符合大根堆的条件
                parent = child;//更新parent下标
                child = 2*parent+1;//更新child下标
            }else {//否则不用换位置
                break;
            }
        }
    }
    //建堆
    public static void createHeap(int[] array) {
        //从小到大排序 -》 大根堆
        for (int i = (array.length-1 - 1) / 2;  i >= 0 ; i--) {
            siftDown(array,i,array.length-1);
        }
    }
    /**
     * 时间复杂度:O(N*logN)  都是这个时间复杂度
     * 复杂度:O(1)
     * 稳定性:不稳定的排序
     * @param array
     */
    public static void heapSort(int[] array) {
        createHeap(array);//O(n)
        int end = array.length-1;//end表示当前末尾元素的下标
        while (end > 0) {//O(N*logN)
            int tmp = array[end];//因为要交换末尾与堆顶元素,所以先缓存末尾元素
            //已经建好堆了,这时堆顶(0下标元素)就是当前的最大值
            array[end] = array[0];//将他提取出来,放到数组的末尾,固定住
            array[0] = tmp;//将末尾元素换到堆顶
            end--;//固定了一个当前堆中的最大值之后,下一次参与排序的元素就得减少一个
            siftDown(array,0,end);//将剩余元素继续变成一个大根堆
        }
    } 
相关文章
|
3天前
|
监控 算法 网络协议
Java 实现局域网电脑屏幕监控算法揭秘
在数字化办公环境中,局域网电脑屏幕监控至关重要。本文介绍用Java实现这一功能的算法,涵盖图像采集、数据传输和监控端显示三个关键环节。通过Java的AWT/Swing库和Robot类抓取屏幕图像,使用Socket进行TCP/IP通信传输图像数据,并利用ImageIO类在监控端展示图像。整个过程确保高效、实时和准确,为提升数字化管理提供了技术基础。
35 15
|
2月前
|
算法 数据处理 C语言
C语言中的位运算技巧,涵盖基本概念、应用场景、实用技巧及示例代码,并讨论了位运算的性能优势及其与其他数据结构和算法的结合
本文深入解析了C语言中的位运算技巧,涵盖基本概念、应用场景、实用技巧及示例代码,并讨论了位运算的性能优势及其与其他数据结构和算法的结合,旨在帮助读者掌握这一高效的数据处理方法。
49 1
|
3月前
|
存储 Java
Java中的HashMap和TreeMap,通过具体示例展示了它们在处理复杂数据结构问题时的应用。
【10月更文挑战第19天】本文详细介绍了Java中的HashMap和TreeMap,通过具体示例展示了它们在处理复杂数据结构问题时的应用。HashMap以其高效的插入、查找和删除操作著称,而TreeMap则擅长于保持元素的自然排序或自定义排序,两者各具优势,适用于不同的开发场景。
55 1
|
9天前
|
缓存 算法 搜索推荐
Java中的算法优化与复杂度分析
在Java开发中,理解和优化算法的时间复杂度和空间复杂度是提升程序性能的关键。通过合理选择数据结构、避免重复计算、应用分治法等策略,可以显著提高算法效率。在实际开发中,应该根据具体需求和场景,选择合适的优化方法,从而编写出高效、可靠的代码。
24 6
|
2月前
|
存储 算法 搜索推荐
Python 中数据结构和算法的关系
数据结构是算法的载体,算法是对数据结构的操作和运用。它们共同构成了计算机程序的核心,对于提高程序的质量和性能具有至关重要的作用
|
20天前
|
存储 缓存 安全
Java 集合江湖:底层数据结构的大揭秘!
小米是一位热爱技术分享的程序员,本文详细解析了Java面试中常见的List、Set、Map的区别。不仅介绍了它们的基本特性和实现类,还深入探讨了各自的使用场景和面试技巧,帮助读者更好地理解和应对相关问题。
37 5
|
2月前
|
数据采集 存储 算法
Python 中的数据结构和算法优化策略
Python中的数据结构和算法如何进行优化?
|
2月前
|
算法
数据结构之路由表查找算法(深度优先搜索和宽度优先搜索)
在网络通信中,路由表用于指导数据包的传输路径。本文介绍了两种常用的路由表查找算法——深度优先算法(DFS)和宽度优先算法(BFS)。DFS使用栈实现,适合路径问题;BFS使用队列,保证找到最短路径。两者均能有效查找路由信息,但适用场景不同,需根据具体需求选择。文中还提供了这两种算法的核心代码及测试结果,验证了算法的有效性。
110 23
|
2月前
|
算法
数据结构之蜜蜂算法
蜜蜂算法是一种受蜜蜂觅食行为启发的优化算法,通过模拟蜜蜂的群体智能来解决优化问题。本文介绍了蜜蜂算法的基本原理、数据结构设计、核心代码实现及算法优缺点。算法通过迭代更新蜜蜂位置,逐步优化适应度,最终找到问题的最优解。代码实现了单链表结构,用于管理蜜蜂节点,并通过适应度计算、节点移动等操作实现算法的核心功能。蜜蜂算法具有全局寻优能力强、参数设置简单等优点,但也存在对初始化参数敏感、计算复杂度高等缺点。
62 20
|
2月前
|
机器学习/深度学习 算法 C++
数据结构之鲸鱼算法
鲸鱼算法(Whale Optimization Algorithm,WOA)是由伊朗研究员Seyedali Mirjalili于2016年提出的一种基于群体智能的全局优化算法,灵感源自鲸鱼捕食时的群体协作行为。该算法通过模拟鲸鱼的围捕猎物和喷出气泡网的行为,结合全局搜索和局部搜索策略,有效解决了复杂问题的优化需求。其应用广泛,涵盖函数优化、机器学习、图像处理等领域。鲸鱼算法以其简单直观的特点,成为初学者友好型的优化工具,但同时也存在参数敏感、可能陷入局部最优等问题。提供的C++代码示例展示了算法的基本实现和运行过程。
57 0

热门文章

最新文章