排序的简单理解(下)

4.交换排序

        基本思想:所谓交换,就是根据序列中两个记录键值的比较结果来对换这两个记录在序列中的位置

        交换排序的特点是:将键值较大的记录向序列的尾部移动,键值较小的记录向序列的前部移动。

4.1 冒泡排序

        冒泡排序(Bubble Sorting)即:通过对待排序的序列从前往后,依次比较相邻元素的值,若发现逆序则交换位置,使较大的元素逐渐移动到后部

4.1.1 算法分析

        下面的分析以将序列{2, 9, 7, 10, 30}从小到大排序为例!

        基本思想就是,在每一趟排序最终实现将该趟得到的最大的数移到序列的最后端

        核心操作就是通过比较相邻两个元素实现当相邻的两个元素逆序的时候,我们就交换它们。

第1趟排序:
        第1趟排序共比较了4次,将最大的数30冒泡到了序列的尾部。

第2趟排序:
        由于第一趟排序已经将最大是数30给冒泡到了最末端,因此在本次排序中,不需要再比较最后一个元素,故此本趟共比较了3次,将子序列(前四个元素)中最大的数10(整个序列中倒数第二大的数)冒泡到了子序列的尾端(原序列的倒数第二个位置)。

第3趟排序:
        在第三趟排序时,同理,倒数两个元素位置已经确定,即第一、第二大的数已经排好位置,只需要再将倒数第三大的数确认即可。故比较2次,实现倒数第三大的数9的位置确定。

第4趟排序:
        在第四趟排序时,只有第一、第二个元素的位置还不确定,只需要比较一次,若逆序,则交换即可。到此,排序算法完成,原序列已经排序成为一个递增的序列!

小结

  • 一共进行了数组大小-1次趟排序,即外层循环arr.length-1次;
  • 每趟排序进行了逐趟减小次数的比较,即内层循环arr.length-i-1次,i从0依次增加。

4.1.2 代码实现

        详细代码如下:

/**
     * @version 1.0
     * 冒泡排序
     */
        public static void main(String[] args) {
            int[] array = {2,9,7, 10, 30};
            //排序前
            System.out.println("排序前:" + Arrays.toString(array));
            //冒泡排序
            for (int i = 0; i < array.length - 1; i++) {
                System.out.println("第" + (i+1) + "趟排序开始!");
                for (int j = 0; j < array.length - i - 1; j++) {
                    //如果前面的数比后面的数大,则交换
                    if(array[j] > array[j+1]){
                        //交换
                        int temp = array[j];
                        array[j] = array[j+1];
                        array[j+1] = temp;
                    }
                    System.out.println("------第" + (j+1) + "趟排序: " + Arrays.toString(array));
                }
                System.out.println("第" + (i+1) + "趟排序完成: " + Arrays.toString(array));
                System.out.println("================================================");
            }

            //输出排序后的结果
            System.out.println("排序后:" + Arrays.toString(array));
        }

        结果展示:

排序前:[2, 9, 7, 10, 30]
第1趟排序开始!
------第1趟排序: [2, 9, 7, 10, 30]
------第2趟排序: [2, 7, 9, 10, 30]
------第3趟排序: [2, 7, 9, 10, 30]
------第4趟排序: [2, 7, 9, 10, 30]
第1趟排序完成: [2, 7, 9, 10, 30]
================================================
第2趟排序开始!
------第1趟排序: [2, 7, 9, 10, 30]
------第2趟排序: [2, 7, 9, 10, 30]
------第3趟排序: [2, 7, 9, 10, 30]
第2趟排序完成: [2, 7, 9, 10, 30]
================================================
第3趟排序开始!
------第1趟排序: [2, 7, 9, 10, 30]
------第2趟排序: [2, 7, 9, 10, 30]
第3趟排序完成: [2, 7, 9, 10, 30]
================================================
第4趟排序开始!
------第1趟排序: [2, 7, 9, 10, 30]
第4趟排序完成: [2, 7, 9, 10, 30]
================================================
排序后:[2, 7, 9, 10, 30]

进程已结束,退出代码0

4.1.3 算法的不足及优化

        我们仔细观察一下我们上述数组在进行冒泡法时每一个外循环后的结果:

                 

        我们发现其实在第一次外循环结束之后,我们的当前数组已经完成了冒泡法整个算法的最终结果,但是由于我们的算法规定,它依旧要跑array.length-1次外循环,故此我们需要完成代码的优化,当我们的数组提前完成排序后,就要提前结束外循环,终止我们的冒泡排序;

        我们可以发现一个无序的数组在经过冒泡算法的排序之后,这些元素的位置在最后都是固定的,每一次的内循环,相应的元素可以理解为都是往那个自己最终的位置上跑,但是当一次内循环之后,发现我们当前的元素位置和该次内循环开始之前的元素位置一样,没有发生变化,这时候我们可以确认我们的元素都已经到达了自己的最终位置,可以提前终止冒泡算法,不在进行其他的外循环;

        所以最终的解决方案就是我们设置一个flag标志位,来判断当前内循环前后数组的元素有没有发生顺序的变化,优化代码如下图所示;

 public static void main(String[] args) {
        int[] array = {5, 1, 2, 3, 4};
        //排序前
        System.out.println("排序前:" + Arrays.toString(array));

        boolean flag = false; //用于标记是否进行了交换,true则说明进行了交换,false表示无

        //冒泡排序
        for (int i = 0; i < array.length - 1; i++) {
            System.out.println("第" + (i+1) + "趟排序开始!");
            for (int j = 0; j < array.length - i - 1; j++) {
                //如果前面的数比后面的数大,则交换
                if(array[j] > array[j+1]){
                    //交换
                    flag = true; //标记进行了交换
                    int temp = array[j];
                    array[j] = array[j+1];
                    array[j+1] = temp;
                }
                System.out.println("------第" + (j+1) + "趟排序: " + Arrays.toString(array));
            }
            System.out.println("第" + (i+1) + "趟排序完成: " + Arrays.toString(array));
            System.out.println("================================================");
            if (!flag){
                //如果没有进行交换则直接退出,说明排序已经完成
                break;
            }else {
                //回退
                flag = false;
            }
        }
        //输出排序后的结果
        System.out.println("排序后:" + Arrays.toString(array));
    }

        测试结果: 

                          

        如图所示,经过优化后,我们相比于之前的代码,减少了两次内循环,提前结束了冒泡算法,大大的节省了时间资源; 

4.1.4 冒泡排序的特性总结

1、冒泡排序是一种非常容易理解的排序

        时间复杂度:O(N^2)

        空间复杂度:O(1)

        稳定性:稳定

2、什么时候最快?
        当输入的数据已经是正序时,我们的内循环和外循环的执行次数比较少;

4.2 快速排序

        快速排序是对冒泡排序的一种改进。基本思想为:通过一趟排序将要排序的数据分割为独立的两个部分,其中一部分的所有数据比另外一部分的所有数据要小,然后按照此方法对这两部分分别进行快速排序,整个过程可以递归进行,以此达到整个数据变成有序序列。

4.2.1 算法分析(递归)

        快速排序算法通过多次比较和交换来实现排序,其排序流程如下:
        (1)首先设定一个分界值,通过该分界值将数组分成左右两部分。
        (2)将大于或等于分界值的数据集中到数组右边,小于分界值的数据集中到数组的左边。此时,左边部分中各元素都小于分界值,而右边部分中各元素都大于或等于分界值。
        (3)然后,左边和右边的数据可以独立排序。对于左侧的数组数据,又可以取一个分界值,将该部分数据分成左右两部分,同样在左边放置较小值,右边放置较大值。右侧的数组数据也可以做类似处理。
        (4)重复上述过程,可以看出,这是一个递归定义。通过递归将左侧部分排好序后,再递归排好右侧部分的顺序。当左、右两个部分各数据排序完成后,整个数组的排序也就完成了。

        下面来举例来详细的分析:

        首先,我们会把数组中的一个数当作基准数,然后从两边进行检索。

        其次,按照如下步骤进行:

                1、先从右边检索比基准数小的
                2、再从左边检索比基准数大的
                3、一旦检索到,就停下,并将检索到的两个元素进行交换
                4、重复上述步骤,直到检索相遇,则替换基准数,并更新区间,递归进行,最终序列会变得有序

        接下来就以{6,1,8,0,2,9,5,3,7}为例详细来分析一下步骤,具体分析一下第一趟排序:以6为基准数的步骤:

1、红色块标识基准数,left、right初始位置如图所示

2、right不断向左移动,寻找比基准数6小的数,如图所示,找到了3

3、此时left开始移动,不断向右移动,寻找比基准数大的数,找到了8,这时,left、right都找到了对应的数,进行交换:

4、right继续向左寻找比基准数6小的数,找到后停止移动,此时left继续向右寻找比基准数大的数,当left与right都找到对应的数后,再次将二者的数值进行交换。

5、重复上述步骤,一直到left与right相遇,二者共同指向了5的位置,则将基准数与该位置的数进行交换,这样就可以观察到,6的左边都是比6小的,右边都是比6大的。

6、该过程需要递归进行,直到序列有序。即以5为基准数,递归6左边的区间,再以9为基数递归6右边的区间,反复进行,直到left >right退出。

4.2.2 代码实现

        冒泡法代码如下:

public static void quickSort(int[] arr, int left, int right) {
            //边界条件
            if (left > right){
                return;
            }

            //定义基准数和左右指针
            int l = left;
            int r = right;
            int base = arr[left];

            //循环,将比基准数小的放在左边,比基准数大的放在右边
            while (l != r){
                //先从右边找比基准数小的,停下
                while (arr[r] >= base && l < r){
                    r--;
                }
                //从左边找比基准数大的,停下
                while (arr[l] <= base && l < r){
                    l++;
                }
                //此时已经找到对应的l 和 r,进行交换
                int temp = arr[l];
                arr[l] = arr[r];
                arr[r] = temp;
            }
            //至此,基准数两边都按照需要排好了,只需要将基准数与lr相遇的位置进行交换
            arr[left] = arr[l];
            arr[l] = base;
            //打印中间结果
            System.out.println(Arrays.toString(arr));
            //先向左找
            quickSort(arr, left, r-1);
            //向右递归
            quickSort(arr, l+1, right);
        }

        测试代码:

public static void main(String[] args) {
     int[] arr = {6,1,8,0,2,9,5,3,7};
     quickSort(arr, 0, arr.length-1);
     System.out.println("排序后: " + Arrays.toString(arr));
}

        测试结果:

4.2.3 快速排序非递归实现  

思路:

  • 建立一个栈

  • 先让一组数据的起点入栈

  • 再让一组数据的终点出栈

  • 然后两次出栈,分别作为该数据的起点与终点

  • 然后经过我们上面所写的方法进行排序后

  • 再将两组数据进行入栈

  • 以此循环直到栈为空

        代码实现: 

    //快速排序递归实现
    public int[] quickSortPlus(int[] array) {
        int[] arr = Arrays.copyOf(array,array.length);
        Deque<Integer> stack = new LinkedList<>();
        int left = 0;
        int right = array.length-1;
        int pivot = 0;
        stack.push(left);
        stack.push(right);
        while (!stack.isEmpty()) {
            right= stack.pop();
            left = stack.pop();
            pivot = partition(arr,left,right);
            if(pivot > left+1) {
                stack.push(left);
                stack.push(pivot-1);
            }
            if(pivot < right-1) {
                stack.push(pivot+1);
                stack.push(right);
            }
        }
        return arr;
    }
    private  int partition(int[] array,int left,int right) {
        int tmp = array[left];
        while (left < right) {
            while (left< right && array[right] >= tmp) {
                right--;
            }
            array[left] = array[right];
            while (left< right && array[left] <= tmp) {
                left++;
            }
            array[right] = array[left];
        }
        array[left] = tmp;
        return left;
    }

4.2.4 特性总结

1. 快速排序整体的综合性能和使用场景都是比较好的,所以才敢叫 快速 排序

2. 时间复杂度: O(N*logN)

3. 空间复杂度:O(logN)

4. 稳定性:不稳定

5. 归并排序

        归并排序(MERGE-SORT)是建立在归并操作上的一种有效的排序算法,该算法是采用分治法(Divide and Conquer)的一个非常典型的应用。将已有序的子序列合并,得到完全有序的序列;即先使每个子序列有序,再使子序列段间有序。若将两个有序表合并成一个有序表,称为二路归并,归并排序核心步骤图解:

5.1算法分析 

合并相邻有序子序列 

...以此类推

...

...

如果当其中的一个子序列全部转移到temp数组中时,另外未空的子序列的元素直接全部按当前顺序移入到temp中即可。

5.2 代码实现

        代码部分:

static int count = 0;
        public static void main(String[] args) {
            int[] arr = {10, 6, 7, 1, 3, 4, 2,9};
            int[] temp = new int[arr.length];
            mergeSort(arr, 0, arr.length - 1, temp);
            System.out.println("归并排序后: arr[] = " + Arrays.toString(arr));
        }

        //归并排序
        public static void mergeSort(int[] arr, int left, int right, int[] temp){
            if (left < right){
                int mid = left - (left - right) / 2;
                //向左递归分解
                mergeSort(arr, left, mid, temp);
                //向右递归分解
                mergeSort(arr, mid + 1, right, temp);
                //排序 合并
                merge(arr, left, mid, right, temp);
            }
        }
        /**
         * 合并的方法
         * @param arr  排序的原始数组
         * @param left  左边有序序列的初始索引
         * @param mid  中间索引
         * @param right  右边索引
         * @param temp  中转数组
         */
        public static void merge(int[] arr, int left, int mid, int right, int[] temp){
            int i = left; //初始化i,左边有序序列的初始索引
            int j = mid + 1; //初始化j,右边有序序列的初始索引
            int t = 0; //指向temp数组的当前索引
            //先把左右两边有序数据按照规则填充到temp数组,直到左右两边有一边处理完毕
            while (i <= mid && j <= right){
                if (arr[i] <= arr[j]){
                    temp[t] = arr[i];
                    t++;
                    i++;
                }else {
                    temp[t] = arr[j];
                    t++;
                    j++;
                }
            }
            //把剩余的一方依次填充到temp数组
            while (i <= mid){ //左边序列还有剩余的元素
                temp[t++] = arr[i++];
            }
            while (j <= right){ //右边序列还有剩余的元素
                temp[t++] = arr[j++];
            }
            //将temp数组的元素拷贝到arr
            //拷贝每次小序列
            t = 0;
            int tempLeft = left;
            while (tempLeft <= right){
                arr[tempLeft++] = temp[t++];
            }
            count++;
            System.out.println("第" + count + "次合并: arr[] = " + Arrays.toString(arr));
        }

        测试结果展示:

        分析:

{10, 6, 7, 1, 3, 4, 2,9}被拆分成了{10, 6}{7, 1}{3, 4}{2, 9}:

第一次合并:{6, 10}有序
第二次合并:{1, 7}有序
第三次合并: {1, 6, 7, 10}有序
第四次合并:{3, 4}有序
第五次合并:{2, 9}有序
第六次合并: { 2,3,4,9}有序
第七次合并:{1,2,3,4,6,7,9,10}有序

5.3 特点总结

  1. 归并的缺点在于需要O(N)的空间复杂度,归并排序的思考更多的是解决在磁盘中的外排序问题。

  2. 时间复杂度:O(N*logN)

  3. 空间复杂度:O(N)

  4. 稳定性:稳定

 5.4 海量数据的排序问题

        外部排序:排序过程需要在磁盘等外部存储进行的排序

        前提:内存只有 1G,需要排序的数据有 100G

        因为内存中因为无法把所有数据全部放下,所以需要外部排序,而归并排序是最常用的外部排序:

         1. 先把文件切分成 200 份,每个 512 M

         2. 分别对 512 M 排序,因为内存已经可以放的下,所以任意排序方式都可以

        3. 进行 2路归并,同时对 200 份有序文件做归并过程,最终结果就有序了

6.排序算法复杂度及稳定性分析

        详解总概图如下所示:

ps:本次的内容就到这里了,本文的相关内容吸取了博主【兴趣使然黄小黄 】的相关思想,如果大家感兴趣的话,可以去了解一下他,如果喜欢的话还请一键三连哦!!!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/238429.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Python图像转PDF神器:教你一步到位实现自动化处理

什么是 img2pdf 库&#xff1f; img2pdf 是一个 Python 库&#xff0c;它可以让你轻松地把多张图像转换为 PDF 文件。它支持多种图像格式&#xff0c;如 JPG, PNG, GIF, BMP 等&#xff0c;并且可以自动调整图像的大小和方向&#xff0c;以适应 PDF 的页面大小和方向。它还可以…

剑指offer(C++)-JZ49:丑数(算法-其他)

作者&#xff1a;翟天保Steven 版权声明&#xff1a;著作权归作者所有&#xff0c;商业转载请联系作者获得授权&#xff0c;非商业转载请注明出处 题目描述&#xff1a; 把只包含质因子2、3和5的数称作丑数&#xff08;Ugly Number&#xff09;。例如6、8都是丑数&#xff0c;…

鸿蒙开发之状态管理@State

1、视图数据双向绑定 鸿蒙开发采用的声明式UI&#xff0c;利用状态驱动UI的更新。其中State被称作装饰器&#xff0c;是一种状态管理的方式。 状态&#xff1a;指的是被装饰器装饰的驱动视图更新的数据。 视图&#xff1a;是指用户看到的UI渲染出来的界面。 之所以成为双向…

量化交易与人工智能:Python库的应用与效用

&#x1f482; 个人网站:【 海拥】【神级代码资源网站】【办公神器】&#x1f91f; 基于Web端打造的&#xff1a;&#x1f449;轻量化工具创作平台&#x1f485; 想寻找共同学习交流的小伙伴&#xff0c;请点击【全栈技术交流群】 量化交易简介 量化交易是一种利用计算机算法执…

gdb使用

> 作者简介&#xff1a;დ旧言~&#xff0c;目前大二&#xff0c;现在学习Java&#xff0c;c&#xff0c;c&#xff0c;Python等 > 座右铭&#xff1a;松树千年终是朽&#xff0c;槿花一日自为荣。 > 目标&#xff1a;熟练掌握gdb的使用 > 毒鸡汤&#xff1a;这个世…

arkts编译报错-arkts-limited-stdlib错误【Bug已完美解决-鸿蒙开发】

文章目录 项目场景:问题描述原因分析:解决方案:适配指导案例此Bug解决方案总结项目场景: arkts编译报错-arkts-limited-stdlib错误。 我用Deveco studio4.0 beta2开发应用,报arkts-limited-stdlib错误 报错内容为: ERROR: ArKTS:ERROR File: D:/prRevivw/3792lapplica…

排序算法:【选择排序]

一、选择排序——时间复杂度 定义&#xff1a;第一趟排序&#xff0c;从整个序列中找到最小的数&#xff0c;把它放到序列的第一个位置上&#xff0c;第二趟排序&#xff0c;再从无序区找到最小的数&#xff0c;把它放到序列的第二个位置上&#xff0c;以此类推。 也就是说&am…

Shell函数数组练习

1、编写函数&#xff0c;实现打印绿色OK和红色FAILED&#xff0c;判断是否有参数&#xff0c;存在为Ok&#xff0c;不存在为FAILED [rootshell ~]# vim ok.sh #!/bin/bash read -p "请输入一个参数:" i function ok…

FFmpeg之AVHWAccel

这也是ffmpeg解码器中比较重要的一个模块&#xff0c;很多人认识它应该是通过一条命令 ffmpeg -hwaccel cuda -hwaccel_output_format cuda -i input.mp4 -c:v h264_nvenc -b:v 5M output.mp4命令地址&#xff1a;英伟达ffmpeg 大家可能觉得这就是nvcodec了&#xff0c;后来发…

域渗透之Exchange

域内部署Exchange 首先这里环境的话是&#xff1a; DC: win2012 exchange服务器: win2012 exchange 2016首先我们去装win2012虚拟机的时候需要给两个网卡&#xff0c;一个是内网&#xff0c;一个是外网的网卡。 内网的dns设置为域控的IP。 外网就不需要指定ip了。 首先需要…

《码农的噩梦与修电脑的奇幻之旅》

故事从一个充满梦想的码农学习计算机编程开始。他对编写程序充满了热情&#xff0c;认为自己就像是一位能够编织魔法的巫师&#xff0c;能够创造出炫酷的虚拟世界。 然而&#xff0c;这个充满幻想的故事在码农入门的第一天就遭遇了突如其来的挫折。电脑故障了&#xff01;所有…

GPT-4V 在机器人领域的应用

在科技的浩渺宇宙中&#xff0c;OpenAI如一颗璀璨的星辰&#xff0c;于2023年9月25日&#xff0c;以一种全新的方式&#xff0c;向世界揭示了其最新的人工智能力作——GPT-4V模型。这次升级&#xff0c;为其旗下的聊天机器人ChatGPT装配了语音和图像的新功能&#xff0c;使得用…

zabbix6入门到精通(2)宏定义

zabbix6入门到精通&#xff08;2&#xff09;宏定义 https://www.yuque.com/fenghuo-tbnd9/ffmkvs/sipmmw https://www.zabbix.com/documentation/6.0/zh/manual/appendix/macros/supported_by_location 配置— 主机 — 主机名称 — {$CPU.INTERVAL.TIME} CPU评估间隔时间…

Qt Desktop Widgets 控件绘图原理逐步分析拆解

Qt 是目前C语言首选的框架库。之所以称为框架库而不单单是GUI库&#xff0c;是因为Qt提供了远远超过GUI的功能封装&#xff0c;即使不使用GUI的后台服务&#xff0c;也可以用Qt大大提高跨平台的能力。 仅就界面来说&#xff0c;Qt 保持各个平台绘图等效果的统一&#xff0c;并…

Linux常用命令---- test 命令

文章目录 基本语法文件测试检查文件是否存在检查文件是否是目录检查文件是否为空检查文件是否可读、可写或可执行 字符串测试检查字符串是否为空检查字符串是否相等检查字符串是否不相等 数字测试检查数字是否相等检查数字是否大于或小于 在Linux操作系统中&#xff0c;test命令…

Oracle 透明网关安装

Oracle 11g透明网关连接Sqlserver oracle 透明网关是oracle连接异构数据库提供的一种技术。通过Gateway&#xff0c;可以在Oracle里透明的访问其他不同的数据库&#xff0c;如SQL Server, DB2, Sybase等等&#xff0c;就像远程Oracle数据库一样。配置后的sql查询的处理流程&…

数据库中常用的锁

目录 1、数据库中常用的锁类型 2、常见的数据库 3、以MySQL为例 3.1 MySQL的事务 3.2 MySQL事务的四大特性 1. 原子性&#xff08;Atomicity&#xff09; 2. 一致性&#xff08;Consistency&#xff09; 3. 隔离性&#xff08;Isolation&#xff09; ⭐mysql中的事务隔…

容器化升级对服务有哪些影响?

容器技术是近几年计算机领域的热门技术&#xff0c;特别是随着各种云服务的发展&#xff0c;越来越多的服务运行在以 Docker 为代表的容器之内。 本文我们就来分享一下容器化技术相关的知识。 容器化技术简介 相比传统虚拟化技术&#xff0c;容器技术是一种更加轻量级的操作…

程序员考公笔记之逻辑判断(图形推理)

文章目录 写在前面1、逻辑判断1.1、图形推理1.1.1、位置类1.1.2、样式类1.1.3、数量类1.1.4、属性类1.1.5、六面体 写在前面 1、逻辑判断 1.1、图形推理 观察&#xff1a;先宏观&#xff0c;再微观 图形推理的命题形式&#xff1a; 一组式 观察路径&#xff1a;顺序看(考最…

数据结构之优先级队列(堆)及top-k问题讲解

&#x1f495;"哪里会有人喜欢孤独&#xff0c;不过是不喜欢失望。"&#x1f495; 作者&#xff1a;Mylvzi 文章主要内容&#xff1a;数据结构之优先级队列(堆) 一.优先级队列 1.概念 我们已经学习过队列&#xff0c;队列是一种先进先出(FIFO)的数据结构&#xff…