0

0

Java面试之常用的排序算法及其复杂度分析

煙雲

煙雲

发布时间:2026-01-06 07:17:12

|

284人浏览过

|

来源于php中文网

原创

冒泡排序因逻辑直观、易暴露边界漏洞而常被面试考察,时间复杂度恒为o(n²),空间复杂度o(1);关键优化是添加swapped标志实现提前终止,使最好情况达o(n)。

java面试之常用的排序算法及其复杂度分析

冒泡排序为什么在面试里总被问,但实际几乎不用

因为它的逻辑最直观,适合考察对「比较-交换」过程的理解,也容易暴露边界处理漏洞。时间复杂度固定是 O(n²),无论最好、最坏、平均情况;空间复杂度是 O(1)(原地排序)。面试手写时,常被忽略的点是提前终止优化——如果某轮没发生任何交换,说明已有序,应直接退出。

public static void bubbleSort(int[] arr) {
    int n = arr.length;
    for (int i = 0; i < n - 1; i++) {
        boolean swapped = false;
        for (int j = 0; j < n - 1 - i; j++) {
            if (arr[j] > arr[j + 1]) {
                int temp = arr[j];
                arr[j] = arr[j + 1];
                arr[j + 1] = temp;
                swapped = true;
            }
        }
        if (!swapped) break; // 关键优化,否则无法达到最好情况 O(n)
    }
}
  • 测试用例务必包含已排序数组,验证 swapped 优化是否生效
  • 注意内层循环上界是 n - 1 - i,不是 n - i,否则可能越界
  • Java 中传入的是数组引用,无需返回值,但需明确说明这是原地修改

快速排序的 partition 实现有哪两种主流写法

面试高频考点:lomutohoare 分区方案。前者以最后一个元素为 pivot,代码简洁但对重复元素不友好;后者用首尾双指针,效率略高且天然更稳定(相同元素分布更均匀),但边界条件易错。两者平均时间复杂度都是 O(n log n),最坏退化到 O(n²)(如已排序数组选端点作 pivot),必须强调随机化 pivot 或三数取中来缓解。

// Lomuto 风格(推荐面试写,逻辑清晰)
private static int partition(int[] arr, int low, int high) {
    int pivot = arr[high]; // 取末尾为 pivot
    int i = low - 1;       // i 指向小于 pivot 的右边界
    for (int j = low; j < high; j++) {
        if (arr[j] <= pivot) {
            i++;
            swap(arr, i, j);
        }
    }
    swap(arr, i + 1, high);
    return i + 1;
}
  • hoare 版本中,两个指针从两端向中间走,首次相遇位置不一定是 pivot 最终位置,需额外判断
  • 递归调用时,lomuto 的左右区间是 [low, pi-1][pi+1, high]hoare[low, pi][pi+1, high],容易混淆
  • Java 8 的 Arrays.sort(int[]) 对基本类型用的是双轴快排(Dual-Pivot Quicksort),不是单 pivot,这点可作为延伸加分项

归并排序如何体现“稳定”和“非原地”的本质特征

稳定性意味着相等元素的相对顺序不变,归并排序天然满足——合并时若 left[i] == right[j],优先取 left[i] 即可。但它需要额外 O(n) 空间存临时数组,不是原地算法。时间复杂度严格是 O(n log n),不受输入数据影响,适合对最坏性能有要求的场景(比如实时系统)。

SekoTalk
SekoTalk

商汤科技推出的AI对口型视频创作工具

下载
private static void mergeSort(int[] arr, int[] temp, int left, int right) {
    if (left >= right) return;
    int mid = left + (right - left) / 2;
    mergeSort(arr, temp, left, mid);
    mergeSort(arr, temp, mid + 1, right);
    merge(arr, temp, left, mid, right);
}
<p>private static void merge(int[] arr, int[] temp, int left, int mid, int right) {
System.arraycopy(arr, left, temp, left, right - left + 1);
int i = left, j = mid + 1, k = left;
while (i <= mid && j <= right) {
if (temp[i] <= temp[j]) { // 注意这里是 <=,保证稳定性
arr[k++] = temp[i++];
} else {
arr[k++] = temp[j++];
}
}
while (i <= mid) arr[k++] = temp[i++];
while (j <= right) arr[k++] = temp[j++];
}
  • 面试手写常漏掉 System.arraycopy 这一步,直接在原数组上合并会导致数据覆盖
  • 递归入口需传入辅助数组 temp,避免每层都 new,否则空间复杂度变成 O(n log n)
  • 堆排序虽也是 O(n log n),但不稳定,且实际性能通常不如优化后的快排或归并

堆排序建堆过程为何是 O(n),而不是直觉上的 O(n log n)

关键在自底向上调整(siftDown)的代价分析:叶子节点不用调整;倒数第二层最多下沉 1 层;倒数第三层最多下沉 2 层……数学推导可得总操作数上限是 2n。而如果用 ninsert(即自顶向下 siftUp),才是 O(n log n)。Java 中没有内置二叉堆类支持索引访问,手写时要注意数组下标从 0 开始时,左子节点是 2*i + 1,不是 2*i

立即学习Java免费学习笔记(深入)”;

  • 建堆后,每次取出最大值(arr[0])与末尾交换,再对剩余部分 siftDown(0),共 n-1 次,每次 O(log n)
  • 堆排序是原地的,但不稳定——相同元素在下沉过程中可能跨过彼此改变顺序
  • 面试若被问“什么时候选堆排序”,答:内存受限 + 要求最坏 O(n log n) + 不关心稳定性(如 Top-K 问题)

实际编码中,除非特定约束,否则直接用 Arrays.sort()。手写排序算法的价值不在替代它,而在暴露你对数据移动、比较次数、内存访问模式的直觉——这些细节,往往比背下复杂度公式更重要。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
sort排序函数用法
sort排序函数用法

sort排序函数的用法:1、对列表进行排序,默认情况下,sort函数按升序排序,因此最终输出的结果是按从小到大的顺序排列的;2、对元组进行排序,默认情况下,sort函数按元素的大小进行排序,因此最终输出的结果是按从小到大的顺序排列的;3、对字典进行排序,由于字典是无序的,因此排序后的结果仍然是原来的字典,使用一个lambda表达式作为key参数的值,用于指定排序的依据。

406

2023.09.04

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

910

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

597

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

294

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

210

2025.08.29

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

431

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

600

2023.08.10

页面置换算法
页面置换算法

页面置换算法是操作系统中用来决定在内存中哪些页面应该被换出以便为新的页面提供空间的算法。本专题为大家提供页面置换算法的相关文章,大家可以免费体验。

485

2023.08.14

Golang 测试体系与代码质量保障:工程级可靠性建设
Golang 测试体系与代码质量保障:工程级可靠性建设

Go语言测试体系与代码质量保障聚焦于构建工程级可靠性系统。本专题深入解析Go的测试工具链(如go test)、单元测试、集成测试及端到端测试实践,结合代码覆盖率分析、静态代码扫描(如go vet)和动态分析工具,建立全链路质量监控机制。通过自动化测试框架、持续集成(CI)流水线配置及代码审查规范,实现测试用例管理、缺陷追踪与质量门禁控制,确保代码健壮性与可维护性,为高可靠性工程系统提供质量保障。

48

2026.02.28

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 4.1万人学习

C# 教程
C# 教程

共94课时 | 10.6万人学习

Java 教程
Java 教程

共578课时 | 75.7万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号