广告
返回顶部
首页 > 资讯 > 后端开发 > Python >Java超详细整理讲解各种排序
  • 950
分享到

Java超详细整理讲解各种排序

2024-04-02 19:04:59 950人浏览 安东尼

Python 官方文档:入门教程 => 点击学习

摘要

目录稳定性直接插入排序希尔排序选择排序堆排序冒泡排序快速排序归并排序计数排序稳定性 两个相等的数据,如果经过排序后,排序算法能保证其相对位置不发生变化,则我们称该算法是具备稳定性的排

稳定性

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

直接插入排序

直接插入排序就是每次选择无序区间的第一个元素,在有序区间内选择合适的位置插入。

数组下标为1开始,将下标为1上的值取出来放在tmp中,然后它和前面的下标j上的值进行比较,如果前面下标j上的值比它大,则前面下标j上的值往后走一步,直到比到j回退到了-1或者j下标上的值比tmp小为止,然后把tmp插入到下标为[j+1]的位置上。然后i就继续往后走,继续比较,直到i走到数组的最后。

代码示例:


    public static void insertSort(int[] array) {
        for(int i=1;i<array.length;i++) {
            int tmp = array[i];
            int j = 0;
            for (j = i - 1; j >= 0; j--) {
                if (tmp < array[j]) {
                    array[j + 1] = array[j];
                } else {
                    break;
                }
            }
            array[j + 1] = tmp;
        }
    }

希尔排序

希尔排序是对直接插入排序的优化。将数据进行相应的分组,分为gap组,然后按照直接插入排序的方法排序。 当gap > 1时都是预排序,目的是让数组更接近于有序。当gap == 1时,数组已经接近有序的了,这样就会很快。这样整体而言,可以达到优化的效果。

    
    public static void shellSort(int[] array) {
        int gap=array.length-1;
        while(gap>1){
            shell(array,gap);
            gap/=2;
        }
        shell(array,1);//保证最后是1组
    }
    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 = i-gap; j >=0; j=j-gap) {
                if(tmp<array[j]){
                    array[j+gap]=array[j];
                }else{
                    break;
                }
            }
            array[j+gap]=tmp;
        }
    }

选择排序

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

代码示例:

 
public static void selectSort(int[] array){
        for(int i=0;i<array.length;i++){
            for(int j=i+1;j< array.length;j++){
                if(array[j]<array[i]){
                    swap(array,i,j);
                }
            }
        }
    }
    public static void swap(int[] array,int i,int j){
        int tmp=array[i];
        array[i]=array[j];
        array[j]=tmp;
    }
    //选择排序还可以找到最小值下标再交换
    public static void selectSort1(int[] array) {
        for (int i = 0; i < array.length; i++) {
            int minIndex = i;
            for (int j = i+1; j < array.length; j++) {
                //找到最小值下标
                if(array[j] < array[minIndex]) {
                    minIndex = j;
                }
            }
            swap(array,i,minIndex);
        }

堆排序

基本原理也是选择排序,只是不在使用遍历的方式查找无序区间的最大的数,而是通过堆来选择无序区间的最大的数。

注意: 排升序要建大堆;排降序要建小堆。 (在堆那里有提过)

代码示例:

 
    public static void heapSort(int[] array){
        creatHeap(array);//先创建堆
        int end=array.length-1;
        //交换后调整(N*log2^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++;
            }
            if(array[child]>array[parent]){
                swap(array,child,parent);
                parent=child;
                child=parent*2-1;
            }else{
                break;
            }
        }
    }

冒泡排序

在无序区间,通过相邻数的比较,将最大的数冒泡到无序区间的最后,持续这个过程,直到数组整体有序。

代码示例:


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

快速排序

1、 从待排序区间选择一个数,作为基准值 (pivot) ;

2、 partition: 遍历整个待排序区间,将比基准值小的(可以包含相等的)放到基准值的左边,将比基准值大的(可以包含相等的)放到基准值的右边;

3、 采用分治思想,对左右两个小区间按照同样的方式处理,直到小区间的长度 == 1 ,代表已经有序,或者小区间的长度 == 0 ,代表没有数据。 代码示例:


//递归方式实现
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--;
            }
            array[start]=array[end];
            while(start<end && array[start]<=tmp){
                start++;
            }
            array[end]=array[start];
        }
        array[start]=tmp;//start==end时,找到了基准点
        return end;
    }

上面代码还有待优化,如果我们是一个有序数组,我们在找基准值进行相应处理的时候可能会出现单分支情况,这时候我们可以让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;
        }
        //优化--找基准前,先找到中间值,三数取中法(防止有序情况下出现单分支)
        int minValIndex=findValINdex(array,left,right);
        swap(array,minValIndex,left);
        int pivot=partition(array,left,right);
        quick(array,left,pivot-1);
        quick(array,pivot+1,right);
    }
   //三数取中 
   private static int findValINdex(int[] array,int start,int end){
        int mid=start+((end-start)>>>1);
        //(start+end)/2
        if(array[start]<array[end]){
            if(array[mid]>array[end]){
                return end;
            } else if (array[mid]<array[start]) {
                return start;
            }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 quickSort(int[] array){
        quick(array,0,array.length-1);
    }
    //递归,找到了基准点,分别再对基准点左边和基准点右边找
    public static void quick(int[] array,int left,int right){
        if(left>=right){
            return;
        }
        //优化--如果区间内的数据,在排序的过程当中,小于某个范围了,可以使用直接插入排序
        if(right-left+1<150){
            insertSort1(array,left,right);
            return;
        }
        //优化--找基准前,先找到中间值,三数取中法(防止有序情况下出现单分支)
        int minValIndex=findValINdex(array,left,right);
        swap(array,minValIndex,left);
        int pivot=partition(array,left,right);
        quick(array,left,pivot-1);
        quick(array,pivot+1,right);
    }
    private static void insertSort1(int[] array,int start,int end){
        for (int i = 1; i <= end; i++) {
            int tmp=array[i];
            int j=i-1;
            for(j=i-1;j>=start;j--){
                if(array[j]>tmp){
                    array[j+1]=array[j];
                }else{
                    //array[j+1]=tmp;
                    break;
                }
            }
            array[j+1]=tmp;
        }
    }

非递归实现:

public static void quickSort1(int[] array){
        int left=0;
        int right=array.length-1;
        Stack<Integer> stack=new Stack<>();
        int pivot=partition(array,left,right);
        //如果左边或者右边只剩下一个数据了,那就已经有序了,不需要在排序
        if(left+1<pivot){
            //左边至少有两个元素
            stack.push(left);
            stack.push(pivot-1);
        }
        if(right-1>pivot){
            //右边至少有两个元素
            stack.push(right);
            stack.push(pivot+1);
        }
        while(!stack.isEmpty()){
            left=stack.pop();
            right=stack.pop();
            pivot=partition(array,left,right);
            if(left+1<pivot){
                //左边至少有两个元素
                stack.push(left);
                stack.push(pivot-1);
            }
            if(right-1>pivot){
                //右边至少有两个元素
                stack.push(right);
                stack.push(pivot+1);
            }
        }
    }

归并排序

归并排序 是建立在归并操作上的一种有效的排序算法 , 将已有序的子序列合并,得到完全有序的序列。即先使每个子序列有序,再使子 序列段间有序。若将两个有序表合并成一个有序表,称为二路归并。

代码示例:


    //递归方式实现
    public static void mergeSort(int[] array){
        mergeSortInternal(array,0,array.length-1);
    }
    public static void mergeSortInternal(int[] array,int left,int right){
        if(left>=right){
            return;
        }
        int mid=left+((right-left)>>>1);
        //mid=(left+right)/2;
        //左边
        mergeSortInternal(array,left,mid);
        //右边
        mergeSortInternal(array,mid+1,right);
        //合并
        merge(array,left,mid,right);
    }
    public static void merge(int[] array,int start,int mid,int end) {
        int s1 = start;
        int e1 = mid;
        int s2 = mid+1;
        int e2 = end;
        int i = 0;
        int[] tmp = new int[array.length];
        while (s1 <= e1 && s2 <= e2) {
            if (array[s1] <=array[s2]) {
                tmp[i] = array[s1];
                //tmp[i++]=array1[s1++];
                i++;
                s1++;
            } else {
                tmp[i] = array[s2];
                i++;
                s2++;
            }
        }
        while(s1<=e1){
            tmp[i++]=array[s1++];
        }
        while(s2<=e2){
            tmp[i++]=array[s2++];
        }
        for (int j = 0; j < i; j++) {
            array[j+start]=tmp[j];
        }
    }

非递归方式--分组归并(先tmp个元素有序然后在合并)

 //tmp:代表组内元素个数
public static void mergeSort1(int[] array){
        int tmp=1;
        while(tmp<array.length-1){
              //数组每次都要进行遍历,确定要归并的区间
            for (int i = 0; i < array.length; i+=tmp*2) {
                int left=i;
                int mid=left+tmp-1;
                //防止越界
                if(mid>=array.length){
                    mid=array.length-1;
                }
                int right=mid+tmp;
                //防止越界
                if(right>=array.length){
                    right=array.length-1;
                }
                //下标确定之后进行合并
                merge(array,left,mid,right);
            }
            tmp=tmp*2;
        }
    }

计数排序

以上排序都是通过比较的排序,接下来介绍一种不需要比较的排序--计数排序。

代码示例:


public static void countingSort(int[] array) {
        int minVal=array[0];
        int maxVal=array[0];
        for(int i=0;i<array.length;i++){
            if(array[i]<minVal){
                minVal=array[i];
            }
            if(array[i]>maxVal){
                maxVal=array[i];
            }
        }
        int[] count=new int[maxVal-minVal+1];//下标默认从0开始
        for (int i=0;i<array.length;i++){
            //统计array数组当中,每个数据出现的次数
            int index=array[i];
            //为了空间的合理使用 这里需要index-minVal  防止623-600
            count[index-minVal]++;
        }
        //说明,在计数数组当中,已经把array数组当中,每个数据出现的次数已经统计好了
        //接下来,只需要,遍历计数数组,把数据写回array
        int indexArray=0;
        for(int i=0;i<count.length;i++){
            while (count[i]>0){
                array[indexArray]=i+minVal;
                count[i]--;
                indexArray++;
            }
        }
    }

到此这篇关于Java超详细整理讲解各种排序的文章就介绍到这了,更多相关Java排序内容请搜索编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程网!

--结束END--

本文标题: Java超详细整理讲解各种排序

本文链接: https://www.lsjlt.com/news/165551.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • Java超详细整理讲解各种排序
    目录稳定性直接插入排序希尔排序选择排序堆排序冒泡排序快速排序归并排序计数排序稳定性 两个相等的数据,如果经过排序后,排序算法能保证其相对位置不发生变化,则我们称该算法是具备稳定性的排...
    99+
    2022-11-13
  • MySql超详细整理讲解各种排序
    目录稳定性直接插入排序希尔排序选择排序堆排序冒泡排序快速排序归并排序计数排序稳定性 两个相等的数据,如果经过排序后,排序算法能保证其相对位置不发生变化,则我们称该算法是具备稳定性的排序算法。 直接插入排序 直接插入排序就...
    99+
    2022-07-29
    MySql排序语句 MySql排序规则
  • Java超详细讲解排序二叉树
    目录排序二叉树概念排序二叉树类的定义添加节点中序遍历查找节点查找某一节点的父节点删除节点排序二叉树概念 二叉排序树(Binary Sort Tree),又称二叉查找树(Binary ...
    99+
    2022-11-13
  • C语言超详细讲解顺序表的各种操作
    目录顺序表是什么顺序表的结构体顺序表的接口函数顺序表相关操作的菜单顺序表的初始化添加元素陈列元素往最后加元素往前面加元素任意位置加元素删除最后元素删除前面元素 删除任意元素...
    99+
    2022-11-13
  • Java 超详细讲解十大排序算法面试无忧
    目录排序算法的稳定性:一.选择排序二.冒泡排序三.插入排序四.希尔排序五.堆排序六.归并排序七.快速排序八.鸽巢排序九.计数排序十.基数排序 排序算法的稳定性:  &nbs...
    99+
    2022-11-13
  • 详细总结各种排序算法(Java实现)
    一、插入类排序1.直接插入排序思想:将第i个插入到前i-1个中的适当位置时间复杂度:T(n) = O(n²)。空间复杂度:S(n) = O(1)。稳定性:稳定排序。如果碰见一个和插入元素相等的,那么插入元素把想插入的元素放在相等元...
    99+
    2023-05-31
    java 排序算法 ava
  • C语言超详细讲解排序算法上篇
    目录1、直接插入排序2、希尔排序(缩小增量排序)3、直接选择排序4、堆排序进入正式内容之前,我们先了解下初阶常见的排序分类 :我们今天讲前四个! 1、直接插入排序 基本思...
    99+
    2022-11-13
  • C语言超详细讲解排序算法下篇
    目录1、冒泡排序2、快速排序 ( 三种方法 )3、归并排序4、排序算法复杂度及稳定性分析 上期学习完了前四个排序,这期我们来学习剩下的三个排序 1、冒泡排序 &n...
    99+
    2022-11-13
  • Java超详细讲解如何生成随机整数
    目录1. java.util.Random2. 数学.随机3. Java 8 Random.ints1. java.util.Random 这Random().nextInt(int...
    99+
    2022-11-13
  • Java超详细讲解异常的处理
    目录1、异常的概念和体系结构1.1异常的概念1.2异常的体系结构及分类2、异常的处理2.1防御式编程2.2异常地抛出2.3异常的捕获(1)异常声明throws(2)try-catch...
    99+
    2022-11-13
  • Java语言中4种内部类的超详细讲解
    目录一.内部类的介绍二.内部类的种类2.1实例内部类      2.2.静态内部类2.3局部内部类2.4匿名内部类总结 一.内部类的...
    99+
    2023-05-17
    java内部类详解 内部类java正确使用 java内部类有哪些
  • Java超详细讲解ArrayList与顺序表的用法
    目录简要介绍Arraylist容器类的使用Arraylist容器类的构造ArrayList的常见方法ArrayList的遍历ArrayList中的扩容机制简要介绍 顺序表是一段物理地...
    99+
    2022-11-13
  • SpringDataJPA实现排序与分页查询超详细流程讲解
    目录前言1、创建持久化实体类2、创建数据访问层3、创建业务层4、创建控制器类5、创建View视图页面6、运行主类 效果如下前言 在实际开发场景中,排序与分页查询是必须的,幸运的是Sp...
    99+
    2022-11-13
    Spring Data JPA排序查询 Spring Data JPA分页查询
  • Java中JDK动态代理的超详细讲解
    目录1. 什么是动态代理?2.动态代理的实现方式有几种?3. JDK动态代理4. CGLB动态代理5.动态代理的效率6.为什么要使用动态代理呢?7. JDK动态代理详细使用介绍总结1...
    99+
    2022-11-13
    Java jdk动态代理 java动态代理原理 jdk动态代理是如何实现的
  • Java 超详细讲解Spring MVC异常处理机制
    目录异常处理机制流程图异常处理的两种方式简单异常处理器SimpleMappingExceptionResolver自定义异常处理步骤本章小结异常处理机制流程图 系统中异常包括两类: ...
    99+
    2022-11-13
  • Java超详细讲解抽象类的原理与用法
    目录1. 抽象类是什么2 抽象类的语法3 抽象类都有什么特性4 抽象类是干什么的1. 抽象类是什么 ️给大家上一篇小作文,看完这个,你就理解了什么叫做抽象类 在面向对象的概念中,所有...
    99+
    2022-11-13
  • Java详细讲解分治算法如何实现归并排序
    目录1.什么是分治算法分治法基本思想2.分治算法的体现——归并排序归并排序基本思想3.代码实现1.什么是分治算法 分治法 分治法,字面意思是“分而治之”,就是把一个复杂的1问题分成两...
    99+
    2022-11-13
  • Java详细讲解堆排序与时间复杂度的概念
    目录一、堆排序1、什么是堆排序2、堆排序思想3、代码实现二、时间复杂度分析1、初始化建堆2、排序重建堆3、总结一、堆排序 1、什么是堆排序 (1)堆排序:堆排序(Heapsort)是...
    99+
    2022-11-13
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作