当前位置: 首页 > article >正文

八种经典排序算法

以下是八种经典排序算法的介绍,包括它们的基本思想、时间复杂度、稳定性以及代码示例:

1. 插入排序

  • 基本思想:每步将一个待排序的元素按其关键码值的大小插入到前面已经排序的部分中,直到全部插入完为止。
  • 时间复杂度:最坏和平均情况为O(n²),最佳情况为O(n)(当数据基本有序时)。
  • 稳定性:稳定的排序方法。
  • 代码示例
     
    public static void insertionSort(int[] array) {  
        int tmp;  
        for (int i = 1; i < array.length; i++) {  
            tmp = array[i];  
            int j = i;  
            for (; j > 0 && array[j - 1] > tmp; j--) {  
                array[j] = array[j - 1];  
            }  
            array[j] = tmp;  
        }  
    }

2. 冒泡排序

  • 基本思想:持续比较相邻的元素,如果第一个比第二个大,就交换它们,直到没有任何一对数字需要比较。
  • 时间复杂度:最坏和平均情况为O(n²),最佳情况为O(n)(当数据已经有序时)。
  • 稳定性:稳定的排序方法。
  • 代码示例
     
    public static void bubbleSort(int[] array) {  
        int tmp;  
        boolean flag;  
        for (int i = array.length - 1; i >= 0; i--) {  
            flag = false;  
            for (int j = 0; j < i; j++) {  
                if (array[j] > array[j + 1]) {  
                    tmp = array[j];  
                    array[j] = array[j + 1];  
                    array[j + 1] = tmp;  
                    flag = true;  
                }  
            }  
            if (!flag) break; // 如果没有发生交换,排序完成  
        }  
    }

3. 选择排序

  • 基本思想:每次从待排序的数据中选出最小(或最大)的元素,存放在序列的起始位置,直到全部排完。
  • 时间复杂度:O(n²)。
  • 稳定性:不稳定的排序方法。
  • 代码示例
     
    public static void selectSort(int[] array) {  
        for (int i = 0; i < array.length - 1; i++) {  
            int min = array[i];  
            int minindex = i;  
            for (int j = i; j < array.length; j++) {  
                if (array[j] < min) {  
                    min = array[j];  
                    minindex = j;  
                }  
            }  
            if (i != minindex) {  
                array[minindex] = array[i];  
                array[i] = min;  
            }  
        }  
    }

4. 希尔排序

  • 基本思想:先取一个小于n的整数作为增量,将数据分组并进行插入排序,逐步减小增量,直到增量为1时进行最后的插入排序。
  • 时间复杂度:依赖于增量序列,通常为O(n log n)到O(n²)之间。
  • 稳定性:不稳定的排序方法。
  • 代码示例
     
    public static void shellSort(int[] array) {  
        int n = array.length;  
        for (int gap = n / 2; gap > 0; gap /= 2) {  
            for (int i = gap; i < n; i++) {  
                int temp = array[i];  
                int j;  
                for (j = i; j >= gap && array[j - gap] > temp; j -= gap) {  
                    array[j] = array[j - gap];  
                }  
                array[j] = temp;  
            }  
        }  
    }

5. 归并排序

  • 基本思想:将数组分成两半,分别对两半进行排序,然后合并已排序的两半。
  • 时间复杂度:O(n log n)。
  • 稳定性:稳定的排序方法。
  • 代码示例
     
    public static void mergeSort(int[] array) {  
        if (array.length < 2) return;  
        int mid = array.length / 2;  
        int[] left = Arrays.copyOfRange(array, 0, mid);  
        int[] right = Arrays.copyOfRange(array, mid, array.length);  
        mergeSort(left);  
        mergeSort(right);  
        merge(array, left, right);  
    }  
    
    private static void merge(int[] array, int[] left, int[] right) {  
        int i = 0, j = 0, k = 0;  
        while (i < left.length && j < right.length) {  
            if (left[i] <= right[j]) {  
                array[k++] = left[i++];  
            } else {  
                array[k++] = right[j++];  
            }  
        }  
        while (i < left.length) array[k++] = left[i++];  
        while (j < right.length) array[k++] = right[j++];  
    }

6. 快速排序

  • 基本思想:选择一个基准元素,将数组分成两部分,左边部分小于基准,右边部分大于基准,然后递归排序这两部分。
  • 时间复杂度:最坏情况为O(n²),平均情况为O(n log n)。
  • 稳定性:不稳定的排序方法。
  • 代码示例
     
    public static void quickSort(int[] array, int low, int high) {  
        if (low < high) {  
            int pivotIndex = partition(array, low, high);  
            quickSort(array, low, pivotIndex - 1);  
            quickSort(array, pivotIndex + 1, high);  
        }  
    }  
    
    private static int partition(int[] array, int low, int high) {  
        int pivot = array[high];  
        int i = low - 1;  
        for (int j = low; j < high; j++) {  
            if (array[j] < pivot) {  
                i++;  
                int temp = array[i];  
                array[i] = array[j];  
                array[j] = temp;  
            }  
        }  
        int temp = array[i + 1];  
        array[i + 1] = array[high];  
        array[high] = temp;  
        return i + 1;  
    }

7. 堆排序

  • 基本思想:利用堆这种数据结构,首先构建一个最大堆,然后将堆顶元素与最后一个元素交换,缩小堆的大小并重新调整堆,直到排序完成。
  • 时间复杂度:O(n log n)。
  • 稳定性:不稳定的排序方法。
  • 代码示例
     
    public static void heapSort(int[] array) {  
        int n = array.length;  
        for (int i = n / 2 - 1; i >= 0; i--) {  
            heapify(array, n, i);  
        }  
        for (int i = n - 1; i > 0; i--) {  
            int temp = array[0];  
            array[0] = array[i];  
            array[i] = temp;  
            heapify(array, i, 0);  
        }  
    }  
    
    private static void heapify(int[] array, int n, int i) {  
        int largest = i;  
        int left = 2 * i + 1;  
        int right = 2 * i + 2;  
        if (left < n && array[left] > array[largest]) {  
            largest = left;  
        }  
        if (right < n && array[right] > array[largest]) {  
            largest = right;  
        }  
        if (largest != i) {  
            int swap = array[i];  
            array[i] = array[largest];  
            array[largest] = swap;  
            heapify(array, n, largest);  
        }  
    }

8. 计数排序

  • 基本思想:通过统计每个元素出现的次数,然后根据计数结果直接将元素放入正确的位置。
  • 时间复杂度:O(n + k),其中n是待排序元素的数量,k是元素值的范围。
  • 稳定性:稳定的排序方法。
  • 代码示例
    public static void countingSort(int[] array) {  
        int max = Arrays.stream(array).max().getAsInt();  
        int[] count = new int[max + 1];  
        for (int num : array) {  
            count[num]++;  
        }  
        int index = 0;  
        for (int i = 0; i < count.length; i++) {  
            while (count[i] > 0) {  
                array[index++] = i;  
                count[i]--;  
            }  
        }  
    }

这些排序算法各有特点,适用于不同的场景和数据规模。在实际应用中,选择合适的排序算法可以显著提高程序的效率。掌握这些算法对于编程和面试都非常重要。


http://www.kler.cn/a/354467.html

相关文章:

  • ORB-SALM3配置流程及问题记录
  • Vue sm3国密 IE模式报错处理
  • Redis数据库笔记——主从复制
  • Three.js教程015:全面讲解Three.js的UV与应用
  • web服务器快速目录搜索遍历工具推荐:Dirsearch
  • Linux好用软件
  • 【Linux】【Jenkins】后端maven项目打包教程-Linux版
  • 在 Android 设备上使用 Kivy 和 OpenCV 实现调用摄像头并显示实时画面
  • Flask创建流式返回的mock脚本
  • Linux 重置 root 密码
  • Flume面试整理-Flume的基本架构
  • 限流是什么?如何限流?怎么限流?
  • 如何轻松使用pip安装Git仓库中的私有Python模块(使用pip和Git仓库发布和安装私有Python模块)
  • 解决ffmpeg通过srt文件给视频添加字幕时乱码问题
  • 【2024最新版】Win10下 Java环境变量配置----适合入门小白
  • RTThread-Nano学习二-RT-Thread启动流程
  • C会赢的!(牛客周赛 Round 58)
  • 力反馈手套如何在VR培训解决方案中为用户提供沉浸式体验?
  • c++链式调用
  • 【css-在一个元素中设置font-size和实际渲染字体大小不一致】
  • CAT(Card Application Toolkit)- LSI
  • Jenkins整合Docker实现CICD自动化部署(若依项目)
  • ESP32-IDF USART 专题
  • 如何在Android中进行日志打印和调试?
  • 即时通讯增加kafka渠道
  • 基于workbox实现PWA预缓存能力