0

0

标题:高效求解多个子数组中位数的算法教程

心靈之曲

心靈之曲

发布时间:2026-01-03 12:13:23

|

233人浏览过

|

来源于php中文网

原创

标题:高效求解多个子数组中位数的算法教程

本文详解如何在大规模数据约束下(n, q ≤ 5×10⁴)快速响应多次子数组中位数查询,重点分析朴素排序法的局限性,并提供可落地的优化思路与正确实现。

在处理子数组中位数查询问题时,核心在于准确理解题设定义:子数组 A[L..R] 的中位数是将其元素升序排列后,位于位置 ⌈len/2⌉(1-indexed)的元素。例如子数组长度为 5,则中位数是第 3 个元素;长度为 4,则是第 2 个元素(非平均值!),即下取整意义上的“左中位数”。这一点极易被误读为偶数长度时取中间两数平均值——但题目明确说明:“ceil(len/2) is called the median”,因此 中位数恒为排序后索引 k = (len - 1) / 2(0-indexed)处的元素

✅ 正确计算中位数索引(0-indexed)

设子数组长度 len = R - L + 1,则 1-indexed 中位位置为 pos = ⌈len/2⌉,对应 0-indexed 索引为:

int k = (len - 1) / 2; // 等价于 (int) Math.ceil(len / 2.0) - 1

例如:

MyMap AI
MyMap AI

使用AI将想法转化为图表

下载
  • len=5 → k = (5-1)/2 = 2 → 第 3 个元素(0-indexed 下标 2)✅
  • len=4 → k = (4-1)/2 = 1 → 第 2 个元素(0-indexed 下标 1)✅
⚠️ 注意:你提供的“尝试代码”中存在多处逻辑错误: median(int N, int[] A) 方法未使用参数 N 实际长度,却对整个 A 排序; getMedian() 递归截断逻辑(Arrays.copyOfRange(A, 1, mid+1))与题目要求的 Q 次任意 [L,R] 查询完全无关; 示例输出 3, 4, 5 对应输入 {2,4,5,3,1,6} 并未说明查询区间,实际应为: Query1: [1,3] → {2,4,5} → sorted → {2,4,5} → median = 4?但示例输出首项是 3 → 实际应为 [1,5]: {2,4,5,3,1} → sorted {1,2,3,4,5} → median = 3; 因此示例隐含查询为 [1,5], [2,4], [3,6] 等,需严格按输入 L,R 提取子数组。

✅ 标准解法(适用于 Q ≤ 5×10⁴ 的务实方案)

由于 N, Q 同阶于 5×10⁴,对每个查询都 Arrays.sort(subarray) 的时间复杂度为 O(Q × len × log len),最坏情况(如每次查整个数组)达 O(Q × N log N) ≈ 5e4 × 5e4 × log(5e4) ≈ 10¹⁰,Java 中必然超时。但若数据无极端卡点,且平均子数组较短,该方法仍可作为基准实现:

import java.util.*;

public class MedianQueries {
    public static void main(String[] args) {
        Scanner sc = new Scanner(System.in);
        int N = sc.nextInt();
        int[] A = new int[N];
        for (int i = 0; i < N; i++) A[i] = sc.nextInt();

        int Q = sc.nextInt();
        while (Q-- > 0) {
            int L = sc.nextInt() - 1; // convert to 0-indexed
            int R = sc.nextInt() - 1;
            int len = R - L + 1;
            int[] sub = Arrays.copyOfRange(A, L, R + 1);
            Arrays.sort(sub);
            int k = (len - 1) / 2; // 0-indexed median position
            System.out.println(sub[k]);
        }
    }
}

? 进阶优化方向(应对严格时限)

若需真正通过 N, Q = 5×10⁴ 的极限数据,应转向以下技术之一:

  • Mo's Algorithm + Balanced BST / Fenwick Tree:离线处理,按块排序查询,维护当前区间内元素频次,二分查找第 k 小值 —— 时间复杂度 O((N + Q) √N log A_max);
  • 整体二分 / 主席树(可持久化线段树):支持在线查询任意区间第 k 小,预处理 O(N log A_max),单次查询 O(log A_max);
  • 随机快速选择(QuickSelect):对每个子数组调用 select(sub, k),期望 O(len),避免全排序,实践中常比 Arrays.sort() 更快。

✅ 总结

  • 中位数定义以 ⌈len/2⌉(1-indexed)为准 → 对应 0-indexed 下标 (len-1)/2;
  • 勿混淆“数学中位数”(偶数长取均值)与本题“竞赛定义中位数”(恒取左中位);
  • 初学推荐先实现朴素 sort + index 版本验证逻辑;
  • 生产级或 OJ 高分需掌握 QuickSelect 或主席树等高级数据结构。

掌握此题,是深入理解区间查询、排序与选择算法边界的良好起点。

热门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

treenode的用法
treenode的用法

​在计算机编程领域,TreeNode是一种常见的数据结构,通常用于构建树形结构。在不同的编程语言中,TreeNode可能有不同的实现方式和用法,通常用于表示树的节点信息。更多关于treenode相关问题详情请看本专题下面的文章。php中文网欢迎大家前来学习。

546

2023.12.01

C++ 高效算法与数据结构
C++ 高效算法与数据结构

本专题讲解 C++ 中常用算法与数据结构的实现与优化,涵盖排序算法(快速排序、归并排序)、查找算法、图算法、动态规划、贪心算法等,并结合实际案例分析如何选择最优算法来提高程序效率。通过深入理解数据结构(链表、树、堆、哈希表等),帮助开发者提升 在复杂应用中的算法设计与性能优化能力。

27

2025.12.22

深入理解算法:高效算法与数据结构专题
深入理解算法:高效算法与数据结构专题

本专题专注于算法与数据结构的核心概念,适合想深入理解并提升编程能力的开发者。专题内容包括常见数据结构的实现与应用,如数组、链表、栈、队列、哈希表、树、图等;以及高效的排序算法、搜索算法、动态规划等经典算法。通过详细的讲解与复杂度分析,帮助开发者不仅能熟练运用这些基础知识,还能在实际编程中优化性能,提高代码的执行效率。本专题适合准备面试的开发者,也适合希望提高算法思维的编程爱好者。

43

2026.01.06

C++高性能网络编程与Reactor模型实践
C++高性能网络编程与Reactor模型实践

本专题围绕 C++ 在高性能网络服务开发中的应用展开,深入讲解 Socket 编程、多路复用机制、Reactor 模型设计原理以及线程池协作策略。内容涵盖 epoll 实现机制、内存管理优化、连接管理策略与高并发场景下的性能调优方法。通过构建高并发网络服务器实战案例,帮助开发者掌握 C++ 在底层系统与网络通信领域的核心技术。

0

2026.03.03

热门下载

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

精品课程

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

共23课时 | 4.1万人学习

C# 教程
C# 教程

共94课时 | 10.6万人学习

Java 教程
Java 教程

共578课时 | 75.9万人学习

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

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