K‘ 未排序数组中的最小/最大元素 |期望线性时间

给定一个不同整数的数组和一个整数,其中 小于数组大小,任务是找到数组中第 k 个最小的元素。

示例:

请注意,解决这个问题的方法有多种,详见《未排序数组中的k个最小/最大元素》。这里讨论的解决方案在实践中效果最佳。

其理念是使用随机枢轴选择来划分数组,通过聚焦子数组(第k个元素必须所在)来减少搜索空间。

一步步的方法:

选择随机枢轴:随机选择一个元素作为枢轴。这有助于避免某些情况下的最坏情况(比如数组已经排序)。

划分:重新排列数组,使所有小于枢轴的元素位于左侧,大于枢轴的元素在右侧。

递归搜索:一旦枢轴定位,如果其索引等于 n 的比较,则它是第 K 大元素。如果没有,则递归搜索相应的划分(左或右),其中 n。

复制代码
// Java program to find K'th Smallest/
// Largest Element in Unsorted Array
import java.util.Random;

class GfG {  
    
    // Partition function: Rearranges elements 
    // around a pivot (last element)
    static int partition(int[] arr, int l, int r) {  
        int x = arr[r];  
        int i = l;     

        // Iterate through the subarray
        for (int j = l; j <= r - 1; j++) {  
            
            // Move elements <= pivot to the left partition
            if (arr[j] <= x) {  
                int temp = arr[i];
                arr[i] = arr[j];
                arr[j] = temp;
                i++; 
            }  
        }  
        
        // Place the pivot in its correct position
        int temp = arr[i];
        arr[i] = arr[r];
        arr[r] = temp;
        return i;  
    }  

    // Randomizes the pivot to avoid worst-case performance
    static int randomPartition(int[] arr, int l, int r) {  
        Random rand = new Random();
        int n = r - l + 1;  
        int pivot = rand.nextInt(n);      
        int temp = arr[l + pivot];
        arr[l + pivot] = arr[r];
        arr[r] = temp;
        return partition(arr, l, r); 
    }  

    // function to find the k'th smallest element using QuickSelect
    static int quickSelect(int[] arr, int l, int r, int k) {  
        
        // Check if k is within the valid range of 
        // the current subarray
        if (k > 0 && k <= r - l + 1) {  
            
            // Partition the array and get the
            // pivot's final position
            int pos = randomPartition(arr, l, r);  

            // If pivot is the k'th element, return it
            if (pos - l == k - 1)  
                return arr[pos];  

            // If pivot's position is larger than k,
            // search left subarray
            if (pos - l > k - 1)  
                return quickSelect(arr, l, pos - 1, k);  

            // Otherwise, search right subarray and adjust k 
            // (k is reduced by the size of the left partition)
            return quickSelect(arr, pos + 1, r, k - (pos - l + 1));  
        }  
        
        // Return infinity for invalid k (error handling)
        return Integer.MAX_VALUE;  
    }

    static int kthSmallest(int[] arr, int k) {
        int n = arr.length;
        return quickSelect(arr, 0, n - 1, k);
    }

    public static void main(String[] args) {  
        int[] arr = {12, 3, 5, 7, 4, 19, 26};  
        int k = 3;  
        System.out.println(kthSmallest(arr, k));  
    }  
}

输出

5

时间复杂度:O(n) 上述解的最坏情况下时间复杂度仍为 O(n2)。在最坏情况下,随机函数可能总是选择一个角元素。然而,平均情况下的时间复杂度为 。分析假设随机数生成器生成输入范围内任意数字的可能性相等。

辅助空间: 自使用常数变量以来,O(1)。O(n)

即使最坏时间复杂度是二次方,这种解在实际中也表现最佳。

复制代码
编程资源
https://pan.quark.cn/s/7f7c83756948
更多资源
https://pan.quark.cn/s/bda57957c548
相关推荐
进击的小头2 小时前
FIR滤波器实战:音频信号降噪
c语言·python·算法·音视频
xqqxqxxq2 小时前
洛谷算法1-1 模拟与高精度(NOIP经典真题解析)java(持续更新)
java·开发语言·算法
razelan2 小时前
初级算法技巧 4
算法
砍树+c+v2 小时前
3a 感知机训练过程示例(手算拆解,代码实现)
人工智能·算法·机器学习
zy_destiny2 小时前
【工业场景】用YOLOv26实现4种输电线隐患检测
人工智能·深度学习·算法·yolo·机器学习·计算机视觉·输电线隐患识别
MengFly_2 小时前
Compose 脚手架 Scaffold 完全指南
android·java·数据库
PPPPickup2 小时前
application.yml或者yaml文件不显示绿色问题
java·数据库·spring
*小海豚*2 小时前
springcloud项目运行启动类无法启动,IDEA也没有任何提示
java·ide
智驱力人工智能2 小时前
货车违规变道检测 高速公路安全治理的工程实践 货车变道检测 高速公路货车违规变道抓拍系统 城市快速路货车压实线识别方案
人工智能·opencv·算法·安全·yolo·目标检测·边缘计算