【java实现+4种变体完整例子】排序算法中【插入排序】的详细解析,包含基础实现、常见变体的完整代码示例,以及各变体的对比表格

以下是插入排序的详细解析,包含基础实现、常见变体的完整代码示例,以及各变体的对比表格:


一、插入排序基础实现

原理

将元素逐个插入到已排序序列的合适位置,逐步构建有序序列。

代码示例
java 复制代码
public class InsertionSort {
    void sort(int[] arr) {
        int n = arr.length;
        for (int i = 1; i < n; i++) {
            int key = arr[i]; // 待插入的元素
            int j = i - 1;
            // 将比 key 大的元素后移
            while (j >= 0 && arr[j] > key) {
                arr[j + 1] = arr[j];
                j--;
            }
            arr[j + 1] = key; // 插入到正确位置
        }
    }
}
复杂度分析
  • 时间复杂度
    • 最坏/平均:O(n²)(逆序或随机数据)。
    • 最好(已有序):O(n)
  • 空间复杂度O(1)
  • 稳定性:稳定(相同值的元素相对顺序不变)。

二、常见变体及代码示例

1. 优化版(减少移动次数)

改进点 :通过减少元素移动的次数,优化内层循环。
适用场景:数据接近有序时效率更高。

java 复制代码
public class OptimizedInsertionSort {
    void sort(int[] arr) {
        int n = arr.length;
        for (int i = 1; i < n; i++) {
            int key = arr[i];
            int j = i - 1;
            // 找到插入位置后一次性移动元素
            while (j >= 0 && arr[j] > key) {
                j--;
            }
            // 将 j+1 到 i 的元素后移一位
            for (int k = i; k > j + 1; k--) {
                arr[k] = arr[k - 1];
            }
            arr[j + 1] = key;
        }
    }
}
2. 二分插入排序

改进点 :用二分查找确定插入位置,减少比较次数。
适用场景:数据规模较大时,减少比较时间。

java 复制代码
public class BinaryInsertionSort {
    void sort(int[] arr) {
        int n = arr.length;
        for (int i = 1; i < n; i++) {
            int key = arr[i];
            int left = 0, right = i - 1;
            // 二分查找插入位置
            while (left <= right) {
                int mid = (left + right) / 2;
                if (arr[mid] > key) {
                    right = mid - 1;
                } else {
                    left = mid + 1;
                }
            }
            // 移动元素并插入
            for (int j = i - 1; j >= left; j--) {
                arr[j + 1] = arr[j];
            }
            arr[left] = key;
        }
    }
}
3. 递归实现

改进点 :用递归替代循环,代码结构更清晰。
适用场景:教学或代码风格偏好递归。

java 复制代码
public class RecursiveInsertionSort {
    void sort(int[] arr, int n) {
        if (n <= 1) return;
        sort(arr, n - 1); // 先排序前 n-1 个元素
        int key = arr[n - 1];
        int j = n - 2;
        // 将比 key 大的元素后移
        while (j >= 0 && arr[j] > key) {
            arr[j + 1] = arr[j];
            j--;
        }
        arr[j + 1] = key;
    }
}

三、变体对比表格

变体名称 时间复杂度 空间复杂度 稳定性 主要特点 适用场景
基础插入排序 O(n²)(平均/最坏), O(n)(最好) O(1) 稳定 简单易实现,适合小规模或部分有序数据 小数据或接近有序的场景
优化版(减少移动次数) O(n²)(平均/最坏), O(n)(最好) O(1) 稳定 减少内层循环的移动次数 数据接近有序时效率提升
二分插入排序 O(n²)(平均/最坏), O(n log n)(比较次数) O(1) 稳定 用二分查找减少比较次数 数据规模较大时减少比较时间
递归实现 O(n²)(所有情况) O(n) 稳定 递归替代循环,代码结构清晰 教学或代码风格偏好递归的场景

四、关键选择原则

  1. 基础场景:优先使用基础实现,因其简单且适用于小规模数据。
  2. 优化需求
    • 接近有序数据:优化版(减少移动次数)可提升效率。
    • 大规模数据:二分插入排序通过减少比较次数优化性能。
  3. 代码风格:递归实现适合教学或偏好函数式编程的场景,但需注意栈空间开销。
  4. 稳定性需求:所有变体均稳定,适用于需要保持元素相对顺序的场景(如排序带键值的记录)。
  5. 极端场景 :已排序数据时,基础实现的时间复杂度降至 O(n),是插入排序的优势场景。

通过选择合适的变体,可在特定场景下优化性能或代码可读性,同时保持算法的稳定性。

相关推荐
naruto_lnq7 小时前
分布式系统安全通信
开发语言·c++·算法
Jasmine_llq7 小时前
《P3157 [CQOI2011] 动态逆序对》
算法·cdq 分治·动态问题静态化+双向偏序统计·树状数组(高效统计元素大小关系·排序算法(预处理偏序和时间戳)·前缀和(合并单个贡献为总逆序对·动态问题静态化
qq_297574677 小时前
【实战教程】SpringBoot 实现多文件批量下载并打包为 ZIP 压缩包
java·spring boot·后端
老毛肚7 小时前
MyBatis插件原理及Spring集成
java·spring·mybatis
学嵌入式的小杨同学8 小时前
【Linux 封神之路】信号编程全解析:从信号基础到 MP3 播放器实战(含核心 API 与避坑指南)
java·linux·c语言·开发语言·vscode·vim·ux
lang201509288 小时前
JSR-340 :高性能Web开发新标准
java·前端·servlet
Re.不晚8 小时前
Java入门17——异常
java·开发语言
爱吃rabbit的mq8 小时前
第09章:随机森林:集成学习的威力
算法·随机森林·集成学习
缘空如是8 小时前
基础工具包之JSON 工厂类
java·json·json切换
追逐梦想的张小年8 小时前
JUC编程04
java·idea