七大排序(Java)

news2024/9/29 17:35:30

目录

一、插入排序

1. 直接插入排序

2. 希尔排序

二、选择排序

1. 直接选择排序

 2. 堆排序

 三、交换排序

1. 冒泡排序

 2. 快速排序

四、归并排序

五、总结


一、插入排序

1. 直接插入排序

   抓一张牌,在有序的牌中,找到合适的位置并且插入。

  • 时间复杂度:O(n^2)
  • 空间复杂度:O(1)
  • 稳定性:稳定

 代码:

    public static void insertSort(long[] array) {
        for (int i = 1; i < array.length; i++) {
            //认为第一个数已经有序,所以循环 n - 1 次
            long tmp = array[i];
            int j = i - 1;
            for(; j >= 0; j--) {
                if (array[j] > tmp) {
                    array[j + 1] = array[j];
                } else {
                    break;
                }
            }
            //j回退到了 小于0 的地方
            array[j + 1] = tmp;
        }
    }

2. 希尔排序

   插入排序的升级版本,即进行大量的分组插排。

  • 时间复杂度:O(n^1.3 ~ n^1.5)
  • 空间复杂度:O(1)
  • 稳定性:不稳定

代码:

    //gap:组数
    public static void shell (int[] array, int gap) {
        for (int i = gap; i < array.length; i++) {
            int tmp = array[i];
            int j = i - gap;
            for (; j >= 0; j = j - gap) {
                if (array[j] > tmp) {
                    array[j + gap] = array[j];
                } else {
                    break;
                }
            }
            array[j + gap] = tmp;
        }
    }

    public static void shellSort(int[] array) {
        int gap = array.length;
        while (gap > 1) {
            shell(array,gap);
            gap = gap / 2;
        }
        shell(array, 1);
    }

二、选择排序

1. 直接选择排序

   找到无序区间中最小的元素的下标,然后将该元素放到无序区间的开始(有序区间的最后)。

  • 时间复杂度:O(n^2)
  • 空间复杂度:O(1)
  • 稳定性:不稳定 

代码:

    // array: 待排序区间
    public static void selectSort(int[] array) {
        for (int i = 0; i < array.length; i++) {
            int minIdx = i;
            for (int j = i + 1; j < array.length; j++) {
                if (array[j] < array[i]) {
                    minIdx = j;
                }
            }
            swap(array, minIdx, i);
        }
    }

 2. 堆排序

   选择排序的升级版本。将无序区间维护成一个大堆(因为从小到达排序,需要大堆)。利用大堆,从无序区间中找到最大值,交换。

  • 时间复杂度:O( N*log(N))
  • 空间复杂度:O(1)
  • 稳定性:不稳定

代码:

    //堆排序
    public static void heapSort(int[] array) {
        //1.建堆 O(n)
        creatHeap(array);
        int end = array.length - 1;
        //2.交换然后调整 O(n * log(n))
        while (end > 0) {
            swap(array, 0, end);
            shiftDown(array, 0, end);
            end--;
        }
    }

    public static void creatHeap(int[] array) {
        for (int parent = (array.length - 1 - 1)/ 2; parent >= 0; parent--) {
            shiftDown(array, parent, array.length);
        }
    }

    public static void shiftDown(int[] array, int parent, int len) {
        int child = 2 * parent + 1; //左孩子下标
        while (child < len) {
            if (child + 1 < len && array[child] < array[child + 1]) {
                child++;
            }
            //child 下标为左右孩子中最大的孩子的下标
            if (array[child] > array[parent]) {
                swap(array, parent, child);
                parent = child;
                child = 2 * parent + 1;
            } else {
                break;
            }
        }
    }

 三、交换排序

1. 冒泡排序

   从前往后,两两元素进行比较,前者大于后者则交换。每次循环完一个元素,后面的有序区间就多加一个元素。

  • 时间复杂度:O(N^2)
  • 有序情况下:O(n)
  • 空间复杂度:O(1)
  • 稳定性:稳定

代码:

    public static void bubbleSort(int[] array) {
        for (int i = 0; i < array.length - 1; i++) {
            boolean flag = false;
            for (int j = 0; j <array.length - 1 - i; j++) {
                if (array[j + 1] < array[j]) {
                    swap(array, j, j + 1);
                    flag = true;
                }
            }
            if (flag == false) {
                break;
            }
        }
    }

 2. 快速排序

   ①在待排序区间内,选择一个基准值(pivot),本次代码中选择的是最右边;

   ②对待排序区间进行 partition 操作,目标:将待排序区间分开,左边的小于等于pivot,右边的大于等于pivot;

   ③分别再对左右两个小区间按照相同的方式进行处理。

  • 时间复杂度:
  •      最好【每次可以均匀的分割待排序序列】:O(N*log(N))
  •      最坏:数据有序 或者逆序的情况  O(N^2)
  • 空间复杂度:
  •      最好:O(logN)
  •      最坏:O(n)   单分支的一棵树
  • 稳定性:不稳定

代码: 

    public static void quickSort(int[] array) {
        quick(array, 0, array.length - 1);
    }

    public static void quick(int[] array, int left, int right) {

        if (left >= right) {
            return;
        }
        int pivot = partition(array, left, right);

        quick(array, left, pivot - 1);
        quick(array, pivot + 1, right);
    }

    //挖坑法
    public static int partition(int[] array, int start, int end) {
        int tmp = array[start];
        while (start < end) {
            while(start < end && array[end] >= tmp) {
                end--;
            }
            //end位置的元素小于基准值,挖坑
            array[start] = array[end];
            while(start < end && array[start] <= tmp) {
                start++;
            }
            //start位置的元素大于基准值,挖坑
            array[end] = array[start];
        }
        array[start] = tmp;
        return start;
    }

   由于用到了递归,当基准值左右边已经有序时,还需要栈空间来进行递归,所以当数据过大时,可能会导致栈溢出的情况,所以要将快速排序进行优化。

  • 当待排序区间元素个数小于某个范围时,可以使用插入排序。
  • 找基准值时,采用三数取中法。

代码: 

public static void quickSort(int[] array) {
        quick(array, 0, array.length - 1);
    }

    public static void quick(int[] array, int left, int right) {
        if (left >= right) {
            return;
        }
        //优化1:元素小于某个区间时,使用插入排序
        if (right - left + 1 < 1400) {
            insertSort(array, left, right);
        }
        //优化2:三数取中法
        int MinIdx = FindMinIdx(array, left, right);
        swap(array, MinIdx, left);

        int pivot = partition(array, left, right);

        quick(array, left, pivot - 1);
        quick(array, pivot + 1, right);
    }

    //挖坑法
    public static int partition(int[] array, int start, int end) {
        int tmp = array[start];
        while (start < end) {
            while(start < end && array[end] >= tmp) {
                end--;
            }
            //end位置的元素小于基准值,挖坑
            array[start] = array[end];
            while(start < end && array[start] <= tmp) {
                start++;
            }
            //start位置的元素大于基准值,挖坑
            array[end] = array[start];
        }
        array[start] = tmp;
        return start;
    }

    //插入排序
    public static void insertSort(int[] array, int start, int end) {
        for (int i = 1; i < end; i++) {
            int tmp = array[i];
            int j = i - 1;
            for (; j >= start; j--) {
                if (array[j] > tmp) {
                    array[j + 1] = array[j];
                } else {
                    //array[j + 1] = tmp
                    break;
                }
            }
            // j 回退到了小于 0 的位置
            //或者从 break 出来
            array[j + 1] = tmp;
        }
    }

    //三数取中法
    private static int FindMinIdx(int[] array, int start, int end) {
        int mid = start + ((end - start) >>> 1);
        if (array[start] < array[end]) {
            if (array[mid] < array[start]) {
                return start;
            } else if (array[mid] > array[end]) {
                return end;
            } else {
                return mid;
            }
        } else {
            if (array[mid] > array[start]) {
                return start;
            } else if (array[mid] < array[end]) {
                return end;
            } else {
                return mid;
            }
        }
    }

    public static void swap(int[] array, int p, int q) {
        int tmp = array[p];
        array[p] = array[q];
        array[q] = tmp;
    }

快速排序非递归版本:利用栈存储 left 和 right 的值。

   public static void quickSort非递归(int[] array) {
        Deque<Integer> stack = new LinkedList<>();
        int left = 0;
        int right = array.length - 1;
        int pivot = partition(array, left, right);

        if (pivot > left + 1) {
            //左边有两位数
            stack.push(left);
            stack.push(pivot - 1);
        }
        if (pivot < right - 1) {
            stack.push(pivot + 1);
            stack.push(right);
        }

        while(!stack.isEmpty()) {
            right = stack.pop();
            left = stack.pop();
            pivot = partition(array, left, right);

            if (pivot > left + 1) {
                //左边有两位数
                stack.push(left);
                stack.push(pivot - 1);
            }
            if (pivot < right - 1) {
                stack.push(pivot + 1);
                stack.push(right);
            }
        }
    }

四、归并排序

   归并排序是采用分治法的一个非常典型的应用。将已有序的子序列合并,得到完全有序的序列;即先使每个子序列有序,再使子序列段间有序。若将两个有序表合并成一个有序表,称为二路归并。

  • 时间复杂度:O(N*log(N))
  • 空间复杂度:O(n)
  • 稳定性:稳定

   在写归并排序代码之前,需要先会写:两个有序数组合并为一个有序数组。

代码:

    //将两个有序数组合并为一个有序数组
    //array1 和 array2 均为有序数组
    public static int[] mergeArray(int[] array1, int[] array2) {
        int[] tmp = new int[array1.length + array2.length];
        int s1 = 0;
        int s2 = 0;
        int e1 = array1.length - 1;
        int e2 = array2.length - 1;
        int k = 0;

        while (s1 <= e1 && s2 <= e2){
            if (array1[s1] <= array2[s2]) {
                tmp[k++] = array1[s1++];
                //k++;
                //s1++;
            } else {
                tmp[k++] = array2[s2++];
            }
        }

        while (s1 <= e1) {
            tmp[k++] = array1[s1++];
        }

        while (s2 <= e2) {
            tmp[k++] = array2[s2++];
        }
        return tmp;
    }

归并排序代码:

    public static void mergeSort(int[] array) {
        mergeSortInternal(array, 0, array.length - 1);
    }

    private static void mergeSortInternal(int[] array, int low, int high) {
        if (low >= high) {
            return;
        }

        int mid = low + ((high - low) >>> 1);

        mergeSortInternal(array, low, mid);
        mergeSortInternal(array, mid + 1, high);

        merge(array, low, mid, high);
    }
   public static void merge(int[] array, int low, int mid, int high) {
        int[] tmp = new int[high - low + 1];
        int k = 0;
        int s1 = low;
        int e1 = mid;
        int s2 = mid + 1;
        int e2 = high;

        while (s1 <= e1 && s2 <= e2) {
            if (array[s1] <= array[s2]) {
                tmp[k++] = array[s1++];
            } else {
                tmp[k++] = array[s2++];
            }
        }

        while (s1 <= e1) {
            tmp[k++] = array[s1++];
        }
        while (s2 <= e2) {
            tmp[k++] = array[s2++];
        }

        //普通的归并数组需要直接返回 tmp 数组
        //但是在归并排序中,要将 tmp 数组重新拷贝到 array 数组中,
        //并且考虑在右子树中合并时,不能覆盖左边已经合并好的元素
        for (int i = 0; i < k; i++) {
            array[low + i] = tmp[i];
        }
    }

非递归版本:

    public static void mergeSort非递归(int[] array) {
        int nums = 1; //每组的数据个数
        while (nums < array.length) {
            for (int i = 0; i < array.length; i++) {
                int left = i;
                int mid = left + nums - 1;
                //防止越界
                if (mid >= array.length) {
                    mid = array.length - 1;
                }
                int right = mid + nums;
                //防止越界
                if (right >= array.length) {
                    right = array.length - 1;
                }
                //下标确定之后进行合并
                merge(array, left, mid, right);
            }
            nums *= 2;
        }
    }

五、总结

   七大排序中:

稳定的排序:冒泡排序插入排序归并排序

插入排序主要看数组是否有序,来影响其时间复杂度
归并排序无论什么情况下,时间复杂度都为 O(N*log(N))

时间复杂度为O(N*log(N))的排序:快速排序堆排序归并排序

要想速度快就选 快速排序
要想稳定就选 归并
要想空间复杂度低就选 堆排序

 

排序时间复杂度空间复杂度稳定性
插入排序O(N^2)O(1)稳定
希尔排序O(N^1.3)O(1)不稳定
选择排序O(N^2)O(1)不稳定
堆排序O(N*log(N))O(1)不稳定
冒泡排序O(N^2)O(1)稳定
快速排序O(N*log(N))O(log(N))不稳定
归并排序O(N*log(N))

O(n)

稳定

 

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

三战阿里测试岗,成功上岸,面试才是测试员涨薪真正的拦路虎...

第一次面试阿里记得是挂在技术面上&#xff0c;当时也是技术不扎实&#xff0c;准备的不充分&#xff0c;面试官出的面试题确实把我问的一头雾水&#xff0c;还没结束我就已经知道我挂了这次面试。 第二次面试&#xff0c;我准备的特别充分&#xff0c;提前刷了半个月的面试题…

防止jar被反编译 不安装jdk运行jar

防止jar被反编译1.pom.xml<repositories><repository><id>jitpack</id><url>https://jitpack.io</url></repository> </repositories><dependencies><dependency><groupId>org.openjfx</groupId><…

RabbitMQ死信队列

目录 一、概念 二、出现死信的原因 三、实战 &#xff08;一&#xff09;代码架构图 &#xff08;二&#xff09;消息被拒 &#xff08;三&#xff09;消息TTL过期 &#xff08;四&#xff09;队列达到最大长度 一、概念 先从概念解释上搞清楚这个定义&#xff0c;死信&…

Spark 3.3.x 读取 HBase 2.x 异常(无法正常连接或读取数据)

无法连接 1. 先检查集群中的 HBase 服务、ZooKeeper 服务是否正常启动&#xff0c;有没有挂掉。 2. Spark 中的 HBase 版本是否与集群一致&#xff0c;代码中的相关包是否导入正确。 3. 连接参数&#xff08;地址、端口&#xff09;是否设置正确&#xff0c;如下所示&#x…

pyqt 制作exe步骤

之前的博客记录 使用pycharmpyqt 编写一个桌面端&#xff08;mac&#xff09;_python开发桌面工具mac_Y_Hungry的博客-CSDN博客 python开发exe程序界面及打包环境配置_Y_Hungry的博客-CSDN博客 1.编写代码 2.打包 pyinstaller -w --add-data "logo.ico;." --add…

Redis常见的数据类型命令

文章目录Redis 常见的数据类型及命令一、常见的NoSQL二、Redis 简介三、key 键的一些操作命令四、Redis的五种基本数据结构1、String&#xff08;字符串&#xff09;介绍常用命令1.1 set/get1.2 append1.3 strlen1.4 setex1.5 mset/mget1.6 setrange/getrange1.7 setnx1.8 incr…

数据库管理工具的使用

目录 摘要 一、Navicat是什么&#xff1f; 二、使用步骤 1.如何下载与安装 2.如何连接远程数据库 总结 摘要 本文主要介绍数据库管理工具的使用 一、Navicat是什么&#xff1f; 它是一款数据库管理工具&#xff0c;将此工具连接数据库,你可以从中看到各种数据库的详细…

JavaScript 面向对象【快速掌握知识点】

目录 类和对象 属性和方法 继承 多态 封装 类和对象 类是用于定义对象的模板或蓝图&#xff1b;它包含对象的属性和方法&#xff0c;我们可以使用class关键字来定义类。 class Person {constructor(name, age) {this.name name;this.age age;}sayHello() {console.log(H…

JSP+SQL电量监视系统设计与实现

技术&#xff1a;Java、JSP等摘要&#xff1a;电脑界常有一股又一股的风潮&#xff0c;在这风潮中&#xff0c;JBuilder 是使用最广泛的工具之一。JBuilder6 提供了可视化集成开发工具&#xff0c;支持开发人员构建标准的爪哇应用系统。其开放的环境、基于组建的开发方式和丰富…

【软件测试面试题】2023年测试人面试专属,最全接口测试面试题大全(附回答)

目录&#xff1a;导读前言一、Python编程入门到精通二、接口自动化项目实战三、Web自动化项目实战四、App自动化项目实战五、一线大厂简历六、测试开发DevOps体系七、常用自动化测试工具八、JMeter性能测试九、总结&#xff08;尾部小惊喜&#xff09;前言 1、HTTP和HTTPS协议…

2022年全国职业院校技能大赛(中职组)网络安全竞赛试题A(9)

目录 竞赛内容 模块A 基础设施设置与安全加固 一、项目和任务描述&#xff1a; 二、服务器环境说明 三、具体任务&#xff08;每个任务得分以电子答题卡为准&#xff09; A-1任务一 登录安全加固&#xff08;Windows&#xff09; 1.密码策略 a.更改或创建密码时执行复杂…

17- TensorFlow实现手写数字识别 (tensorflow系列) (项目十七)

项目要点 模型创建: model Sequential()添加卷积层: model.add(Dense(32, activationrelu, input_dim100)) # 第一层需要 input_dim添加dropout: model.add(Dropout(0.2))添加第二次网络: model.add(Dense(512, activationrelu)) # 除了first, 其他层不要输入shape添加输出…

这是一款值得开发人员认真研究的软件,数据库优化,应用服务器安全优化...

1.查询数据库死锁相关信息2.查看数据库的链接情况3.当前实例上的所有用户4.创建数据库独立密码5.查看数据库使用的端口号6.当前数据库设置的最大连接数7.当前数据库最大的理论可连接数8.当前数据库实例的连接数9.当前数据库连接数10.当前数据库连接超时设置11.当前sqlserver 超…

SMART系统—考试监控及阅卷模块的设计与开发

技术&#xff1a;Java、JSP等摘要&#xff1a;Smart在线考试评估系统(简称“SMART系统”)&#xff0c;是基于Browser/Server&#xff08;简称B/S&#xff09;结构的数据库访问模式&#xff0c;采用Struts Spring Hibernate作为平台搭建的框架开发的一套新型智能的远程教育软件…

伺服三环控制深层原理解析

我们平时使用的工业伺服,通常是成套伺服,即驱动器和电机型号存在配对关系。 但有些时候,我们要用电机定转子和编码器制作非成套电机,这种时候,我们需要对驱动器进行各种设置才能驱动电机。 此篇文章将通过介绍伺服控制的三环控制原理入手来说明我们调试非成套伺服时需要…

2023年微信小程序获取手机号授权登录注册详细教程,包含服务端教程

前言 小程序中有很多地方都会用到用户的手机号&#xff0c;比如登陆注册&#xff0c;填写收货地址等等。有了这个组件可以快速获取微信绑定手机号码&#xff0c;无须用户填写。网上大多数教程还是往年的&#xff0c;而微信官方的api已做了修改。本篇文章将使用最新的方法获取手…

【unity学习记录】Canvas Group组件

&#x1f497; 未来的游戏开发程序媛&#xff0c;现在的努力学习菜鸡 &#x1f4a6;本专栏是我关于游戏开发的学习笔记 &#x1f236;本篇是unity的Canvas Group组件 Canvas Group画布组介绍详解1. Alpha2. Interactable3. Blocks Raycasts4. Ignore Parent Groups介绍 画布组…

用反射模拟IOC模拟getBean

IOC就是spring的核心思想之一&#xff1a;控制反转。这里不再赘述&#xff0c;看我的文章即可了解&#xff1a;spring基础思想IOC其次就是java的反射&#xff0c;反射机制是spring的重要实现核心&#xff0c;今天我看spring的三级缓存解决循坏引用的问题时&#xff0c;发现一个…

机器学习——模型评估

在学习得到的模型投放使用之前&#xff0c;通常需要对其进行性能评估。为此&#xff0c;需使用一个“测试集”(testing set&#xff09;来测试模型对新样本的泛化能力&#xff0c;然后以测试集上的“测试误差( tootino error)作为泛化误差的近似。我们假设测试集是从样本真实分…

ShardingSphere水平、垂直分库、分表和公共表

目录一、ShardingSphere简介二、ShardingSphere-分库分表1、垂直拆分&#xff08;1&#xff09;垂直分库&#xff08;2&#xff09;垂直分表2、水平拆分&#xff08;1&#xff09;水平分库&#xff08;2&#xff09;水平分表三、水平分库操作1、创建数据库和表2、配置分片的规则…