【Java数据结构】排序

【Java数据结构】排序

  • 一、排序
      • 1.1 排序的概念
      • 1.2 排序的稳定性
      • 1.3 内部排序和外部排序
        • 1.3.1 内部排序
        • 1.3.2 外部排序
  • 二、插入排序
      • 2.1 直接插入排序
      • 2.2 希尔排序
  • 三、选择排序
      • 3.1 选择排序
      • 3.2 堆排序
  • 四、交换排序
      • 4.1 冒泡排序
      • 4.2 快速排序
        • Hoare法:
        • 挖坑法:
        • 前后指针:
      • 4.3 快速排序的优化
        • 4.3.1 三数取中法
        • 4.3.2 递归到小的子区间时,使用直接插入排序
      • 4.4 非递归快速排序
  • 五、归并排序
      • 5.1 递归归并排序
      • 5.2 非递归归并排序
  • 六、总结

博客最后附有整篇博客的全部代码!!!

一、排序

1.1 排序的概念

排序是计算机科学中一个非常基础且重要的概念,它指的是将一组对象按照某种顺序排列的过程。排序算法是实现排序功能的具体方法,通过对数据进行比较、交换或移动等操作,使数据元素按照指定的顺序排列。

1.2 排序的稳定性

稳定性是一个重要的概念,它描述了排序算法是否能够保持相同元素的相对顺序不变。

排序稳定性:
一个排序算法是稳定的,如果在排序过程中,两个具有相同键值(或值)的元素在排序前后的相对顺序保持不变。换句话说,如果元素A和B在排序前满足A在B之前,并且它们的键值相同,那么排序后A仍然在B之前。

例如:
下面一组数据,里面有两个相同的值‘8’(为了展现它们的相对位置,我们将两个相同值的‘8’用不同元素表示出来)。在排序之前‘8’
在这里插入图片描述

1.3 内部排序和外部排序

1.3.1 内部排序

内部排序是指在排序过程中,所有待排序的数据都能同时存储在内存中进行处理。由于内存访问速度较快,内部排序通常效率较高,但受限于内存容量,适用于数据量较小的场景。

特点:
存储:所有数据存储在内存中。
效率:通常较快,因为内存访问速度快。
适用场景:数据量较小(通常在几万甚至几十万以内)。

1.3.2 外部排序

外部排序是指待排序的数据量过大,无法全部加载到内存中,需要借助外存(如磁盘)来完成排序的过程。外部排序通常涉及将数据分块处理,排序后再合并。

特点:
存储:数据主要存储在外存(如磁盘),内存仅用于存储部分数据。
效率:通常较慢,因为外存访问速度远低于内存。
适用场景:数据量巨大(如数百万甚至数十亿条记录),无法全部加载到内存中。

二、插入排序

2.1 直接插入排序

思想:

将待排序的元素逐个插入到已经排好序的序列中,从而逐步扩展有序序列的长度,直到所有元素都被插入,整个序列变为有序。

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

代码:

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

2.2 希尔排序

思想:

希尔排序(Shell Sort)是插入排序的一种改进版本,通过将数组分成多个子序列进行排序,逐步缩小子序列的间隔(增量),最终使整个数组有序。
核心思想如下:
选择增量序列:初始增量通常为数组长度的一半,随后逐步减小,直到增量为1。
分组排序:按照当前增量将数组分成多个子序列,对每个子序列进行插入排序。
逐步缩小增量:重复上述过程,直到整个数组基本有序,最后使用增量为1的插入排序完成最终排序。

时间复杂度:O(n^1.3 )至 O( n^1.5)之间。
空间复杂度:O(1)。
稳定性:不稳定。

代码:

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

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

三、选择排序

3.1 选择排序

思想:

选择排序是一种简单直观的排序算法。它的核心思想是:

  1. 每次从未排序的部分中选择最小(或最大)的元素,并将其放到已排序部分的末尾。
  2. 重复上述过程,直到所有元素都被排序。

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

代码:

 public void selectSort(int[] array){
        int minIndex=0;
        for (int i=0;i<array.length;i++){
            minIndex=i;
            for(int j=i+1;j<array.length;j++){
                if(array[minIndex]>array[j]){
                    minIndex=j;
                }
            }
            int tmp=array[i];
            array[i]=array[minIndex];
            array[minIndex]=tmp;
        }
    }

延伸
思路:

定义两个变量‘minIndex’和‘maxIndex’来接收遍历完一遍数组得到的最大值和最小值下标,将得到的最大值和最小值下标分别与这组数组的最左边和最右边的值交换,以此类推。

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

代码:

    public void selectSort2(int[] array) {
        int left=0;
        int right=array.length-1;
        int len=array.length;
        while(left<right){
            int minIndex=left;
            int maxIndex=left;
            for (int i = left; i < len; i++) {
                if(array[i]<array[minIndex]){
                    minIndex=i;
                }
                if(array[i]>array[maxIndex]){
                    maxIndex=i;
                }
            }
            swap(array,left,minIndex);
            left++;
            swap(array,right,maxIndex);
            right--;
            len--;
        }

    }
    public void swap(int[] array,int x,int y){
        int tmp=array[x];
        array[x]=array[y];
        array[y]=tmp;
    }

3.2 堆排序

思路:

建立大根堆,将大根堆的堆顶元素和最后一个元素进行交换,交换完后将剩下的堆重新调整,以此类推。

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

代码:

public void heapSort(int[] array){
        //创建堆
        createHeap(array);
        int end=array.length-1;
        while(end>0){
            swap(array,0,end);
            shiftDown(array,0,end);
            end--;
        }
    }
    public void createHeap(int[] array){
        for(int parent=(array.length-1-1)/2;parent>=0;parent--){
            shiftDown(array,parent,array.length);
        }
    }
    public void shiftDown(int[] array,int parent,int length){
        int child =parent*2+1;
        while(child<length){
            //如果孩子存在,找到左右孩子中较小的孩子,用child标记
            if (child + 1 < length && array[child] < array[child+1]) {
                child++;
            }
            if(array[parent]<=array[child]){
                swap(array,parent,child);
                parent=child;
                child=parent*2+1;
            }else{
                break;
            }
        }
    }

四、交换排序

4.1 冒泡排序

思想:

核心思想是通过相邻元素之间的比较和交换,逐步将最大(或最小)的元素“冒泡”到数组的末尾(或开头)。这个过程重复进行,直到整个数组有序。

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

代码:

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

4.2 快速排序

思想:

核心思想是通过分区操作将数组分为两部分,使得一部分的所有元素都小于(或等于)另一部分的所有元素,然后递归地对这两部分进行排序。

时间复杂度:O(N*logN)至O(n^2)。
空间复杂度:O(logN)至O(N)。
稳定性:不稳定。

Hoare法:

思路:

选定序列第一个为基准,从后边往前找到比基准小的停下来,从前边找到比基准大的停下来,交换直到左右相遇,相遇下标的值与基准交换。

代码:

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

    public void hoareSort(int[] array,int start,int end) {
        if(start>=end){
            return;
        }
        int pivot = partition(array, start, end);
        hoareSort(array,start,pivot-1);
        hoareSort(array,pivot+1,end);
    }

    private int partition(int[] array, int left, int right) {
        int flag=left;
        while(left<right){
          while(left<right&&array[right]>=array[flag]){
                right--;
            }
            while(left<right&&array[left]=<array[flag]){
                left++;
            }
            swap(array,left,right);
        }
        swap(array,left,flag);
        return left;
    }
挖坑法:

思路:

选定序列第一个为基准,从后往前找,找到小于基准的,将其放到基准的位置,接下来从前往后找,找到比基准大的,放到先前后面找到的比基准小的位置,以此类推。

代码:

    private int partition2(int[] array, int left, int right) {
        int flag=array[left];
        while(left<right){
            while(left<right&&array[right]>=flag){
                right--;
            }
            array[left]=array[right];
            while(left<right&&array[left]<=flag){
                left++;
            }
            array[right]=array[left];
        }
        array[left]=flag;
        return left;
    }
前后指针:

思路:

选定序列第一个为基准,定义两个prev和cur来标记下标,遍历序列,满足cur下标的值小于基准,并且prev++下标和cur下标不在同一个下标,交换prev和cur下标的值,如果不满足条件,cur++。

代码:

    private int partition3(int[] array, int left, int right) {
        int prev=left;
        int cur=prev+1;
        while(cur<=right){
           if(array[cur]<array[left]&&array[++prev]!=array[cur]){
               swap(array,prev,cur);
           }
           cur++;
        }
        swap(array,prev,left);
        return prev;
    }

4.3 快速排序的优化

4.3.1 三数取中法

优点:

优化性能:通过选择中值作为基准值,减少了因数据分布不均匀而导致的性能退化。
提高稳定性:在处理接近有序或完全逆序的数据时,性能更加稳定。

代码:

 public void hoareSort(int[] array,int start,int end) {
        if(start>=end){
            return;
        }
        int midIndex=getNumber(array,start,end);
        swap(array,start,midIndex);

        int pivot = partition(array, start, end);
        hoareSort(array,start,pivot-1);
        hoareSort(array,pivot+1,end);
    }
    private int getNumber(int[] array,int left,int right){
        int mid=(left+right)/2;
        if(array[left]<array[right]){
            if(array[mid]<array[left]){
                return left;
            } else if (array[mid]>array[right]) {
                return right;
            }else{
                return mid;
            }
        }else{
            if(array[mid]<array[right]){
                return right;
            } else if (array[mid]>array[left]) {
                return left;
            }else{
                return mid;
            }
        }
    }

4.3.2 递归到小的子区间时,使用直接插入排序

优点:

因为直接插入排序的特点就是越有序越快。

代码:

    public void hoareSort(int[] array,int start,int end) {
        if(start>=end){
            return;
        }
//        int midIndex=getNumber(array,start,end);
//        swap(array,start,midIndex);

        if (end - start+1 <= 10) {
            insertSortRange(array,start,end);
            return;
        }
        int pivot = partition(array, start, end);
        hoareSort(array,start,pivot-1);
        hoareSort(array,pivot+1,end);
    }

    private void insertSortRange(int[] array, int start, int end) {
        int tmp=0;
        for(int i=start+1;i<=end;i++){
            tmp=array[i];
            int j=i-1;
            for(;j>=start;j--){
                if(tmp<array[j]){
                    array[j+1]=array[j];
                }else{
                    break;
                }
            }
            array[j+1]=tmp;
        }
    }

4.4 非递归快速排序

思想:

  1. 通过挖坑法先求得基准下标
  2. 通过基准下标将基准左右两边序列的start和end存进栈中,存入栈中的先后顺序要一致
  3. 通过pop()弹出start和end,再通过挖坑法求得基准下标,以此类推,当栈为空,则证明已经排序好了

代码:

    public void quickNor(int[] array,int start,int end) {
        Stack<Integer> stack=new Stack<>();
        int pivot = partition2(array, start, end);
        if(pivot>start+1){
            stack.push(start);
            stack.push(pivot-1);
        }
        if(pivot<end-1){
            stack.push(pivot+1);
            stack.push(end);
        }
        while(!stack.isEmpty()){
            end=stack.pop();
            start=stack.pop();
            pivot=partition2(array, start, end);
            if(pivot>start+1){
                stack.push(start);
                stack.push(pivot-1);
            }
            if(pivot<end-1){
                stack.push(pivot+1);
                stack.push(end);
            }
        }
    }

五、归并排序

5.1 递归归并排序

思路:

归并排序是一种分治排序算法,其核心思想是将数组分成多个小部分,分别对这些小部分进行排序,然后逐步合并这些有序部分,最终得到一个完全有序的数组。

时间复杂度:O(N*logN)。
空间复杂度:O(N)。
稳定性:稳定。

代码:

    public void mergeSort(int[] nums) {
        mergeSortSplit(nums, 0, nums.length - 1);
    }

    private void mergeSortSplit(int[] nums, int left, int right) {
        if (left >= right) {
            return;
        }
        //分解
        int mid = (left + right) / 2;
        mergeSortSplit(nums, left, mid-1);
        mergeSortSplit(nums, mid + 1, right);
        //合并
        merge(nums, left, mid, right);
    }

    private void merge(int[] nums, int left, int mid, int right) {
        int[] tmp = new int[right - left + 1];
        int k=0;
        int s1=left;
        int s2=mid + 1;
        while(s1<=mid&&s2<=right){
            if(nums[s1]<=nums[s2]){
                tmp[k++]=nums[s1++];
            }else{
                tmp[k++]=nums[s2++];
            }
        }
        while(s1<=mid){
            tmp[k++]=nums[s1++];
        }
        while(s2<=right){
            tmp[k++]=nums[s2++];
        }
        for(int i=left; i<k; i++){
            nums[i]=tmp[i];
        }
    }

5.2 非递归归并排序

    public void mergeSortNor(int[] array){
        int gap=1;
        while(gap<array.length){
            for(int i=0;i<array.length;i=i+gap*2){
                int left=i;
                int mid=left+gap-1;
                if(mid>=array.length){
                    mid=array.length-1;
                }
                int right=mid+gap;
                if(right>=array.length){
                    right=array.length-1;
                }
                merge(array,left,mid,right);
            }
            gap*=2;
        }
    }

六、总结

在这里插入图片描述
此篇博客的全部代码!!!

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

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

相关文章

20250118-读取并显示彩色图像以及提取彩色图像的 R、G、B 分量

读取并显示彩色图像以及提取彩色图像的 R、G、B 分量 import cv2 # 彩图 R、G、B 的提取 import numpy as np from PIL import Image from matplotlib import pyplot as plt1. 读取并显示彩色图像的三种方法&#xff1a; img_path "./data/yndx"1.1 使用 PIL 读取…

【总结盘点类】2024,一场关于海量数据治理以及合理建模的系列写作

目录 1.今年的创作路线 2.先说第一条线 2.1.由日志引出的海量文本数据存储和分析问题 2.2.监控以及监控的可视化 2.3.数据量级再往上走牵扯出了大数据 2.4.由大数据牵扯出的JAVA线程高级内容 3.第二条线&#xff0c;也是2025要继续的主线 1.今年的创作路线 今年的写作内…

SpringBoot整合ES及简单API使用

1、pom文件导入ES依赖 <dependency><groupId>org.elasticsearch.client</groupId><artifactId>elasticsearch-rest-high-level-client</artifactId><version>7.4.2</version> </dependency>2、编写配置&#xff0c;给容器中注…

Ardupilot开源无人机之Geek SDK进展2024-2025

Ardupilot开源无人机之Geek SDK进展2024-2025 1. 源由2. 状态3. TODO3.1 【进行中】跟踪目标框3.2 【暂停】onnxruntime版本3.3 【完成】CUDA 11.8版本3.4 【完成】pytorch v2.5.1版本3.5 【未开始】Inference性能3.6 【未开始】特定目标集Training 4. Extra-Work4.1 【完成】C…

计算机网络 (54)系统安全:防火墙与入侵检测

前言 计算机网络系统安全是确保网络通信和数据不受未经授权访问、泄露、破坏或篡改的关键。防火墙和入侵检测系统&#xff08;IDS&#xff09;是维护网络系统安全的两大核心组件。 一、防火墙 定义与功能 防火墙是一种用来加强网络之间访问控制的特殊网络互联设备&#xff0c;它…

大模型GUI系列论文阅读 DAY1:《基于大型语言模型的图形用户界面智能体:综述》(6.6W 字长文)

摘要 图形用户界面&#xff08;Graphical User Interfaces, GUIs&#xff09;长期以来一直是人机交互的核心&#xff0c;为用户提供了直观且以视觉为驱动的方式来访问和操作数字系统。传统上&#xff0c;GUI交互的自动化依赖于基于脚本或规则的方法&#xff0c;这些方法在固定…

12位磁编码器AS5600磁式转动位置传感器

AS5600 12位磁式转动位置传感器 描述主要优点和特点应用方框图引脚分配引脚描述工作条件详细说明IC 电源管理IC 接口支持的模式寄存器描述阶跃响应和过滤器设置方向&#xff08;顺时针与逆时针&#xff09;磁滞现象磁体检测低功耗模式看门狗定时器 应用信息 描述 智能非接触式电…

微服务知识——4大主流微服务架构方案

文章目录 1、微服务聚合模式2、微服务共享模式3、微服务代理模式4、微服务异步消息模式 微服务是大型架构的必经之路&#xff0c;也是大厂重点考察对象&#xff0c;下面我就重点详解4大主流微服务架构方案。 1、微服务聚合模式 微服务聚合设计模式&#xff0c;解决了如何从多个…

在视频汇聚平台EasyNVR平台中使用RTSP拉流的具体步骤

之前有用户反馈&#xff0c;在EasyNVR平台中添加Pull时使用海康设备的RTSP流地址无法播放。经过研发的优化及一系列严谨的验证流程&#xff0c;我们已确认优化后的EasyNVR平台&#xff0c;通过Pull方式添加海康设备的RTSP流已经能够正常播放。以下是具体的操作步骤&#xff1a;…

【重庆市乡镇界】面图层shp格式arcgis数据乡镇名称和编码wgs84坐标无偏移内容测评

标题中的“最新重庆市乡镇界面图层shp格式arcgis数据乡镇名称和编码wgs84坐标无偏移最新”指的是一个地理信息系统&#xff08;GIS&#xff09;的数据集&#xff0c;特别设计用于ArcGIS软件。这个数据集包含了重庆市所有乡镇的边界信息&#xff0c;以Shapefile&#xff08;.shp…

Linux系统 C/C++编程基础——使用make工具和Makefile实现自动编译

ℹ️大家好&#xff0c;我是练小杰&#xff0c;今天周二了&#xff0c;距离除夕只有&#xff16;天了&#xff0c;新的一年就快到了&#x1f606; 本文是有关Linux C/C编程的make和Makefile实现自动编译相关知识点&#xff0c;后续会不断添加相关内容 ~~ 回顾:【Emacs编辑器、G…

系统思考—转型

我们大多数问题的来源是&#xff1a;人们的思考方式与大自然的运作方式之间的差异。——葛雷果利贝特森&#xff08;人类学家、生物学家及系统思考先驱&#xff09; 在企业转型的过程中&#xff0c;许多企业创始人常常面临一个困境——过去的成功经验和旧有的思维方式&#xf…

【Linux系统】—— 编译器 gcc/g++ 的使用

【Linux系统】—— 编译器 gcc/g 的使用 1 用 gcc 直接编译2 翻译环境2.1 预处理&#xff08;进行宏替换&#xff09;2.2 编译&#xff08;生成汇编&#xff09;2.3 汇编&#xff08;生成机器可识别代码&#xff09;2.4 链接2.5 记忆小技巧2.6 编译方式2.7 几个问题2.7.1 如何理…

LARGE LANGUAGE MODELS ARE HUMAN-LEVEL PROMPT ENGINEERS

题目 大型语言模型是人类级别的提示工程师 论文地址&#xff1a;https://arxiv.org/abs/2211.01910 项目地址&#xff1a;https://github.com/keirp/automatic_prompt_engineer 摘要 通过对自然语言指令进行调节&#xff0c;大语言模型 (LLM) 显示了作为通用计算机的令人印象深…

计算机的错误计算(二百一十八)

摘要 大模型能确定 sin(2.6^100) 的符号吗&#xff1f;实验表明&#xff0c;大模型给的结论是正确的&#xff0c;但其证明过程是错误百出。大模型的推理实在是不敢恭维。 就同样题目&#xff0c;测试一下另外一个大模型。 例1. 能确定 sin(2.6^100) 的符号吗&#xff1f; 下…

51c~SLAM~合集1

我自己的原文哦~ https://blog.51cto.com/whaosoft/12327374 #GSLAM 自动驾驶相关~~~ 一个通用的SLAM架构和基准 GSLAM&#xff1a;A General SLAM Framework and Benchmark 开源代码&#xff1a;https://github.com/zdzhaoyong/GSLAM SLAM技术最近取得了许多成功&am…

【2024年终总结】我与CSDN的一年

&#x1f449;作者主页&#xff1a;心疼你的一切 &#x1f449;作者简介&#xff1a;大家好,我是心疼你的一切。Unity3D领域新星创作者&#x1f3c6;&#xff0c;华为云享专家&#x1f3c6; &#x1f449;记得点赞 &#x1f44d; 收藏 ⭐爱你们&#xff0c;么么哒 文章目录 …

Day 14 卡玛笔记

这是基于代码随想录的每日打卡 226. 翻转二叉树 给你一棵二叉树的根节点 root &#xff0c;翻转这棵二叉树&#xff0c;并返回其根节点。 示例 1&#xff1a; 输入&#xff1a;root [4,2,7,1,3,6,9] 输出&#xff1a;[4,7,2,9,6,3,1]示例 2&#xff1a; 输入&#xff1a;r…

51c大模型~合集105

我自己的原文哦~ https://blog.51cto.com/whaosoft/13101924 #刚刚&#xff0c;ChatGPT开始有了执行力&#xff01; 现在 AI 智能体可以 24*7 小时为你打工。 2025 刚过去了半个月&#xff0c;OpenAI 在智能体领域「开大」了。 今天&#xff0c;OpenAI 正在为 ChatGPT 推出…

《Effective Java》学习笔记——第2部分 对象通用方法最佳实践

文章目录 第2部分 所有对象通用方法一、前言二、最佳实践内容1. equals()方法2. hashCode()方法3. toString() 方法4. clone() 方法5. finalize() 方法6. compareTo()方法&#xff08;实现 Comparable 接口&#xff09; 三、小结 第2部分 所有对象通用方法 一、前言 《Effect…