Java—排序1

本篇将详细讲解插入排序、希尔排序和堆排序三种经典排序算法,包括算法原理、执行过程、易错点分析,并为每种算法提供三道例题及详细解析。


一、插入排序(Insertion Sort)

算法原理

插入排序的核心思想是将待排序数组分为已排序未排序两部分。初始时,已排序部分仅包含第一个元素。随后,从未排序部分取出一个元素,在已排序部分从后向前扫描,找到合适的位置插入该元素(比取出元素大或小时),直到所有元素有序。

  • 时间复杂度:最优O(n)(数组已有序),最差O(n^2)(数组逆序),平均O(n^2)。
  • 空间复杂度$O(1)(原地排序)。
执行过程
  1. 从第二个元素开始遍历(索引i = 1)。
  2. 将当前元素arr[i]暂存为key
  3. j = i - 1向前扫描已排序部分:
    • arr[j] > key,则将arr[j]后移一位。
    • 否则跳出循环。
  4. key插入到正确位置。
易错点
  1. 边界条件:循环索引从1开始,内层循环终止条件为j >= 0
  2. 元素移动:需用while而非for循环,确保及时终止扫描。
  3. 稳定性:遇到相等元素时不移动,保证稳定性。

例题1:对整数数组升序排序
public void insertionSort(int[] arr) { for (int i = 1; i < arr.length; i++) { int key = arr[i]; int j = i - 1; while (j >= 0 && arr[j] > key) { // 易错点:需同时检查边界和大小 arr[j + 1] = arr[j]; j--; } arr[j + 1] = key; } }

解析:经典实现,注意内层循环需同时判断j >= 0arr[j] > key

例题2:对链表排序(使用插入排序)
public ListNode insertionSortList(ListNode head) { ListNode dummy = new ListNode(0); ListNode cur = head; while (cur != null) { ListNode prev = dummy; while (prev.next != null && prev.next.val < cur.val) { prev = prev.next; } ListNode next = cur.next; cur.next = prev.next; prev.next = cur; cur = next; } return dummy.next; }

解析:链表需使用虚拟头节点dummy简化插入操作,注意断开原节点的链接。


例题3:对浮点数数组降序排序
public void insertionSortDesc(double[] arr) { for (int i = 1; i < arr.length; i++) { double key = arr[i]; int j = i - 1; while (j >= 0 && arr[j] < key) { // 降序:将 > 改为 < arr[j + 1] = arr[j]; j--; } arr[j + 1] = key; } }

解析:仅需将内层循环条件改为arr[j] < key即可实现降序。


二、希尔排序(Shell Sort)

算法原理

希尔排序是插入排序的改进版,通过分组插入提升效率。设定一个增量序列(如$n/2, n/4, \dots, 1$),对每个增量间隔的分组进行插入排序,最后增量减至1时整体排序。

  • 时间复杂度:约O(n^1.3)(依赖增量序列)。
  • 空间复杂度:O(1)。
执行过程
  1. 选择初始增量gap = n/2
  2. 对每个增量分组执行插入排序。
  3. 缩小增量(如gap /= 2),重复步骤2,直到gap = 1
易错点
  1. 增量序列:常用Knuth序列(h = 3h + 1)而非简单折半。
  2. 分组方式:每个分组是间隔gap的子序列,而非连续子数组。
  3. 终止条件:增量需覆盖到gap = 1

例题1:整数数组升序排序(Knuth增量)
public void shellSort(int[] arr) { int n = arr.length; int gap = 1; while (gap < n / 3) gap = 3 * gap + 1; // Knuth序列:1, 4, 13, ... while (gap >= 1) { for (int i = gap; i < n; i++) { int key = arr[i]; int j = i; while (j >= gap && arr[j - gap] > key) { // 注意索引 j-gap arr[j] = arr[j - gap]; j -= gap; } arr[j] = key; } gap /= 3; // 缩小增量 } }

解析:使用Knuth序列优化效率,内层循环索引需以gap为步长移动。


例题2:字符串按长度排序
public void shellSortStrings(String[] arr) { int n = arr.length; int gap = 1; while (gap < n / 3) gap = 3 * gap + 1; while (gap >= 1) { for (int i = gap; i < n; i++) { String key = arr[i]; int j = i; while (j >= gap && arr[j - gap].length() > key.length()) { arr[j] = arr[j - gap]; j -= gap; } arr[j] = key; } gap /= 3; } }

解析:比较条件改为字符串长度,注意稳定性不受影响。


例题3:自定义对象按字段排序
class Person { String name; int age; } public void shellSortPersons(Person[] arr) { int n = arr.length; int gap = 1; while (gap < n / 3) gap = 3 * gap + 1; while (gap >= 1) { for (int i = gap; i < n; i++) { Person key = arr[i]; int j = i; while (j >= gap && arr[j - gap].age > key.age) { arr[j] = arr[j - gap]; j -= gap; } arr[j] = key; } gap /= 3; } }

解析:对自定义对象按age字段排序,需确保比较逻辑正确。


三、堆排序(Heap Sort)

算法原理

堆排序基于二叉堆数据结构:

  1. 建堆:将无序数组构建成最大堆(父节点值 ≥ 子节点)。
  2. 排序:交换堆顶(最大值)与末尾元素,缩小堆范围,重新调整堆结构。
  • 时间复杂度:建堆O(n),排序O(n log n),总O(n \log n)。
  • 空间复杂度:O(1)。
执行过程
  1. 建堆:从最后一个非叶子节点(索引n/2 - 1)开始,自底向上执行heapify
  2. 排序
    • 交换堆顶与末尾元素。
    • 堆大小减1,对堆顶执行heapify
    • 重复直到堆大小为1。
易错点
  1. 索引计算:父子节点索引关系为left = 2*i + 1right = 2*i + 2
  2. 堆调整范围:每次交换后堆大小减1,调整范围需随之缩小。
  3. 堆调整方向heapify需从根节点向下递归调整。

例题1:整数数组升序排序
public void heapSort(int[] arr) { int n = arr.length; // 建堆(从最后一个非叶子节点开始) for (int i = n / 2 - 1; i >= 0; i--) { heapify(arr, n, i); } // 排序:交换堆顶与末尾元素 for (int i = n - 1; i > 0; i--) { swap(arr, 0, i); heapify(arr, i, 0); // 调整剩余堆 } } private void heapify(int[] arr, int n, int i) { int largest = i; int left = 2 * i + 1; int right = 2 * i + 2; if (left < n && arr[left] > arr[largest]) largest = left; if (right < n && arr[right] > arr[largest]) largest = right; if (largest != i) { swap(arr, i, largest); heapify(arr, n, largest); // 递归调整子树 } }

解析:注意建堆时需从n/2 - 1开始倒序调整。


例题2:数组降序排序(最小堆)
public void heapSortDesc(int[] arr) { int n = arr.length; // 建最小堆:将比较改为 < for (int i = n / 2 - 1; i >= 0; i--) { heapifyMin(arr, n, i); } // 排序逻辑相同 for (int i = n - 1; i > 0; i--) { swap(arr, 0, i); heapifyMin(arr, i, 0); } } private void heapifyMin(int[] arr, int n, int i) { int smallest = i; int left = 2 * i + 1; int right = 2 * i + 2; if (left < n && arr[left] < arr[smallest]) smallest = left; if (right < n && arr[right] < arr[smallest]) smallest = right; if (smallest != i) { swap(arr, i, smallest); heapifyMin(arr, n, smallest); } }

解析:将heapify中的比较改为<即可实现最小堆,从而得到降序序列。


例题3:对二维数组按行首元素排序
public void heapSort2D(int[][] matrix) { int n = matrix.length; // 建堆(按每行第一个元素) for (int i = n / 2 - 1; i >= 0; i--) { heapify2D(matrix, n, i); } for (int i = n - 1; i > 0; i--) { swapRows(matrix, 0, i); heapify2D(matrix, i, 0); } } private void heapify2D(int[][] matrix, int n, int i) { int largest = i; int left = 2 * i + 1; int right = 2 * i + 2; if (left < n && matrix[left][0] > matrix[largest][0]) largest = left; if (right < n && matrix[right][0] > matrix[largest][0]) largest = right; if (largest != i) { swapRows(matrix, i, largest); heapify2D(matrix, n, largest); } }

解析:比较逻辑改为matrix[i][0],交换时需整行交换(swapRows方法需自定义)。


总结

  1. 插入排序:小规模数据高效,实现简单但$O(n^2)$复杂度。
  2. 希尔排序:通过分组插入提升效率,增量序列影响性能。
  3. 堆排序:O(n log n)复杂度且原地排序,适合大规模数据。

三种排序各有适用场景,理解其原理及易错点对编写正确代码至关重要。

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

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

相关文章

结合温升测试验证工业用PCB线宽电流对照表

温升实测揭秘&#xff1a;工业PCB走线到底能扛多大电流&#xff1f;从一个烧断的铜箔说起某天&#xff0c;一位工程师在调试一台工业变频器时发现&#xff0c;设备运行十几分钟后突然停机。检查发现&#xff0c;主板上一条看似“足够宽”的电源走线竟然局部碳化、断裂——而这根…

手把手教程:搭建AUTOSAR基础软件平台

从零搭建AUTOSAR基础软件平台&#xff1a;实战指南与核心原理深度剖析 你有没有遇到过这样的场景&#xff1f; 一个项目刚做完&#xff0c;客户突然提出&#xff1a;“能不能把这套控制逻辑移植到另一款MCU上&#xff1f;”你打开代码一看——满屏的寄存器操作、硬编码的CAN报…

一文说清JLink驱动安装无法识别的核心要点

一文讲透J-Link驱动装不上、认不出的底层逻辑与实战修复 你有没有遇到过这种情况&#xff1a; 手头项目正紧&#xff0c;调试关键时刻插上J-Link&#xff0c;结果设备管理器里只显示“未知设备”或带黄叹号的USB设备&#xff1f; Keil连不上&#xff0c;Ozone报错&#xff0…

51单片机入门项目:实现LED闪烁的核心要点

从零点亮一盏灯&#xff1a;51单片机LED闪烁实战全解析你有没有过这样的经历&#xff1f;翻开一本嵌入式教材&#xff0c;第一行代码就是P1 0xFE;&#xff0c;然后告诉你“现在P1.0口的LED亮了”。可你心里却满是问号&#xff1a;为什么写个寄存器灯就亮了&#xff1f;电平是怎…

初学51单片机必做项目:Keil流水灯代码超详细版解析

从点亮第一盏灯开始&#xff1a;51单片机流水灯实战全解析你有没有过这样的经历&#xff1f;手握开发板&#xff0c;烧录完程序&#xff0c;却只等来一片死寂——LED一动不动。那一刻的挫败感&#xff0c;我太懂了。当年我第一次写流水灯代码时&#xff0c;连P1 0xFE;这行简单…

hbuilderx开发微信小程序:实战案例从零实现

用 HBuilderX 开发微信小程序&#xff1a;从零搭建一个可上线的实战项目 你有没有遇到过这种情况&#xff1a;想快速做一个微信小程序&#xff0c;但官方开发者工具写代码太“原始”&#xff0c;没有智能提示、不支持 Git、UI 设计也费劲&#xff1f;更头疼的是&#xff0c;一…

2026武汉做网站TOP8盘点:企业数字化解决方案推荐

2026武汉企业建站&#xff1a;数字化转型的核心选择逻辑2026年&#xff0c;武汉中小微企业占市场主体超90%&#xff0c;外贸企业依托长江经济带加速跨境布局&#xff0c;本地商家在消费升级中寻求线上突围。武汉做网站不仅是搭建网页&#xff0c;更是企业数字化的“基础设施”—…

盘式电机 maxwell 电磁仿真模型 双转单定结构,halbach 结构,双定单转 24 槽...

盘式电机 maxwell 电磁仿真模型 双转单定结构&#xff0c;halbach 结构&#xff0c;双定单转 24 槽 20 极&#xff0c;18槽 1 2 极&#xff0c;18s16p&#xff08;可做其他槽极配合&#xff09; 参数化模型&#xff0c;内外径&#xff0c;叠厚等所有参数均可调整 默认模型仅作学…

Keil5 MDK安装教程:新手入门必看的环境准备清单

Keil5 MDK安装实战指南&#xff1a;从零搭建嵌入式开发环境 你是不是刚接触STM32&#xff0c;打开电脑准备写第一行代码时&#xff0c;却被“Keil怎么装&#xff1f;”、“为什么编译报错&#xff1f;”、“下载不了程序怎么办&#xff1f;”这些问题卡住&#xff1f;别急——…

8位加法器硬件连接与调试实战案例

从理论到板级&#xff1a;8位加法器硬件实战中的那些“坑”与突破你有没有遇到过这样的情况——明明逻辑设计完全正确&#xff0c;Verilog代码综合无误&#xff0c;仿真波形也完美匹配真值表&#xff0c;可一旦烧进FPGA、接上拨码开关和数码管&#xff0c;输出就开始乱跳&#…

大学生移动端作业学习数据分析程序设计与实现 微信小程序PHP_nodejs_vue+uniapp

文章目录移动端作业学习数据分析程序设计摘要系统设计与实现的思路主要技术与实现手段源码lw获取/同行可拿货,招校园代理 &#xff1a;文章底部获取博主联系方式&#xff01;移动端作业学习数据分析程序设计摘要 该设计基于微信小程序平台&#xff0c;整合PHP、Node.js、Vue.j…

Keil uVision5调试环境搭建:手把手操作指南

从零搭建Keil uVision5调试环境&#xff1a;工程师的实战手记最近接手一个基于STM32F4的工业控制项目&#xff0c;客户要求在两周内完成Bootloader开发和通信协议联调。时间紧、任务重&#xff0c;第一件事就是——先把调试环境搭稳。别小看这一步。我见过太多团队因为“下载失…

AD23导出Gerber文件的完整示例演示

从设计到制造&#xff1a;在AD23中正确导出Gerber文件的实战全解析你有没有遇到过这样的情况&#xff1f;PCB布局布线完成&#xff0c;DRC全绿&#xff0c;信心满满地导出Gerber发给厂家打样&#xff0c;结果三天后收到回复&#xff1a;“阻焊层缺失”、“钻孔偏移”、“丝印压…

扶贫助农系统及农副产品销售商城系统小程序的实现PHP_nodejs_vue+uniapp

文章目录扶贫助农系统及农副产品销售商城系统小程序的实现系统设计与实现的思路主要技术与实现手段源码lw获取/同行可拿货,招校园代理 &#xff1a;文章底部获取博主联系方式&#xff01;扶贫助农系统及农副产品销售商城系统小程序的实现 该系统采用前后端分离架构&#xff0c…

51单片机核心外设知识点总结:GPIO、按键、中断、定时器与PWM

一、GPIO&#xff1a;单片机与外界交互的基础接口GPIO&#xff0c;全称General Purpose Input Output&#xff0c;即通用输入输出&#xff0c;是单片机与外界进行数据交互的最基本形式。每个GPIO引脚都可以独立配置为输入模式或输出模式&#xff0c;满足不同场景的使用需求。1.…

档案馆参观预约系统 微信小程序PHP_nodejs_vue+uniapp

文章目录档案馆参观预约系统技术架构功能模块设计关键技术实现数据安全与性能优化用户体验设计系统设计与实现的思路主要技术与实现手段源码lw获取/同行可拿货,招校园代理 &#xff1a;文章底部获取博主联系方式&#xff01;档案馆参观预约系统技术架构 该系统采用前后端分离架…

W5500与STM32结合的看门狗机制设计:操作指南

W5500与STM32协同看门狗设计&#xff1a;从原理到实战的完整指南你有没有遇到过这样的场景&#xff1f;一台部署在工厂角落的工业网关&#xff0c;连续运行了几周后突然“失联”——Ping不通、数据中断&#xff0c;但现场检查却发现设备电源正常&#xff0c;MCU似乎还在跑代码。…

React Router严重漏洞可用于访问或修改服务器文件

聚焦源代码安全&#xff0c;网罗国内外最新资讯&#xff01;编译&#xff1a;代码卫士安全研究员在 React Router 中发现多个严重漏洞&#xff0c;可导致攻击者通过目录遍历访问或修改服务器文件。这些漏洞影响 React Router 生态系统中的多个程序包&#xff0c;且CVSS v3评分为…

2025小结:从RL到Agentic RL

如果说训练大模型的目标是稳定且高效地scale up&#xff0c;那Agentic RL给人的印象往往是既不稳定&#xff0c;也不高效&#xff0c;对于资源有限的团队来说&#xff0c;也很难scale。上半年发生过很多变故耽误了不少时间&#xff0c;中间有几个月完全断层&#xff0c;基本没有…

捏着鼻子玩过PEM电解槽模拟的都懂,三维两相流这玩意儿能把人整懵。不过别慌,今天咱们用COMSOL搞点接地气的操作,先来瞅瞅多孔介质这货怎么折腾

PEM电解槽三维两相流模拟&#xff0c;包括电化学&#xff0c;两相流传质&#xff0c;析氢析氧&#xff0c;化学反应热等多物理场耦合&#xff0c;软件comsol&#xff0c;可分析多孔介质传质&#xff0c;析氢析氧过程对电解槽电流密度分布&#xff0c;氢气体积分数&#xff0c;氧…