在线性级别时间内找出无序序列中的第k个元素

在一个无序序列中找出第k个元素,对于k很小或者很大时可以采取特殊的方法,比如用堆排序来实现 。但是对于与序列长度N成正比的k来说,就不是一件容易的事了,可能最容易想到的就是先将无序序列排序再遍历即可找出第k个元素。由于任何基于比较的排序算法不可能用少于Θ(N lgN)次比较来实现将所有元素排序,所以采用排序的方法的时间复杂度是线性对数级别的。

我们可以借鉴快速排序中将序列划分的思想来实现平均情况下线性级别的算法,算法实现如下:

 1 public class KthElement {
 2
 3     private static void exch(Comparable[] a, int i, int j)
 4     {
 5         Comparable swap = a[i];
 6         a[i] = a[j];
 7         a[j] = swap;
 8     }
 9
10     private static boolean less(Comparable a, Comparable b)
11     {
12         return a.compareTo(b) < 0;
13     }
14
15     private static int  partition(Comparable[] a, int lo, int hi)
16     {
17         int i = lo;
18         int j = hi + 1;
19         Comparable v = a[lo];
20         while(true)
21         {
22             while(less(a[++i], v)) if(i == hi) break;
23             while(less(v, a[--j]));
24             if(i >= j) break;
25             exch(a, i, j);
26         }
27         exch(a, lo, j);
28         return j;
29     }
30
31     public static Comparable select(Comparable[] a, int k)
32     {
33         int lo = 0;
34         int hi = a.length - 1;
35         while(hi > lo)
36         {
37             int j = partition(a, lo, hi);
38             if(j == k) break;
39             else if(j > k) hi = j - 1;
40             else if(j < k) lo = j + 1;
41         }
42         return a[k];
43     }
44     public static void main(String[] args) {
45         Integer[] ints = {5, 3, 1, 4, 2};
46         int find = (int) select(ints, 2);
47         System.out.println(find);
48     }
49
50 }

在select方法中,使用partition方法将序列划分。如果k = j,问题就已经解决了; 如果k < j,就继续切分左字数组(令 hi = j - 1);如果k > j,就继续切分右子数组(令lo = j + 1)。该循环保证了lo左边的元素都小于等于a[lo...hi], 而hi右边的元素都大于等于a[lo...hi],我们不断切分直到数组中只剩下第k个元素。为何这个的时间复杂度是线性级别的,证明很复杂,在此给出理想情况下的简易证明。假设每次切分都从中间切分,则所有的比较次数为(N + N/2 + N/4 + N/8 +...)直到找到k,很显然这个和小于2N。平均情况下的复杂度为Θ( 2N + 2Kln(N/K) + 2(N - K)ln(N/(N-K)) )。当K = N/2时,复杂度为Θ((2 + 2ln2)N)。

时间: 2024-11-07 20:11:51

在线性级别时间内找出无序序列中的第k个元素的相关文章

[PY3]——找出一个序列中出现次数最多的元素/collections.Counter 类的用法

问题 怎样找出一个序列中出现次数最多的元素呢? 解决方案 collections.Counter 类就是专门为这类问题而设计的, 它甚至有一个有用的 most_common() 方法直接给了你答案 collections.Counter 类 1. most_common(n)统计top_n from collections import Counter words = [ 'look', 'into', 'my', 'eyes', 'look', 'into', 'my', 'eyes', 't

算法题:找出同一个序列中的最大值和最小值

package arithmetic; /** * 同时找出一个序列中最大值和最小值 * 分两种情况:(1)序列只有两个元素 * (2)序列有多个元素,有多个元素分别从序列的两端开始查找 * @author SHI */ public class MaxAndMin { public static void main(String[] args) { int[] a = { 122, 41, 4, 5, 7, 2, 89, 122, 34, 56 }; int low = 0; int high

算法试题 - 找出一个序列中出现频率最高的三个数

题目 找出一个序列中出现频率最高的三个数 解析 思路一 创建一个新字典, k 为 序列的值, 然后 v 的初始值 0, 然后循环序列进行计数, 然后进行新字典的处理..... 不行, 不好处理了. 此思路不行 思路二 利用 colletctions 的 Counter 模块, 内部有个方法可以解决 k 值问题 答案 答案一 可以往下继续实现, 但是有点麻烦了 li = [2, 5, 3, 4, 1, 8, 1, 2, 6, 6, 1, 5, 1, 5] d = dict.fromkeys(li,

封装函数,找出数组 arr 中重复出现过的元素.

<!DOCTYPE html><html lang="en"><head> <meta charset="UTF-8"> <title>Title</title></head><body><script> //封装函数,找出数组 arr 中重复出现过的元素. //示例: console.log(duplicates( [1, 2, 4, 4, 3, 3, 1,

算法导论:快速找出无序数组中第k小的数

题目描述: 给定一个无序整数数组,返回这个数组中第k小的数. 解析: 最平常的思路是将数组排序,最快的排序是快排,然后返回已排序数组的第k个数,算法时间复杂度为O(nlogn),空间复杂度为O(1).使用快排的思想,但是每次只对patition之后的数组的一半递归,这样可以将时间复杂度将为O(n). 在<算法导论>有详细叙述 这里主要用C++实现,实现思路就是先选取当前数组的第一个数作为"主轴",将后面所有数字分成两部分,前面一部分小于"主轴",后面一部

给定一个字符串,找出这个字符串中首先出现K次的字符

public static Character FindFirstKChar(String str,int k){ String s = str.replaceAll(" ","");         Map<Character, Integer> charCountMap = new HashMap<Character, Integer>();         Character ch = null;         for(int i =

算法:找出 n 个数中最小的 k 个数

最简单的方法是将n个元素排序,取出最小的k个元素.这个算法的时间复杂度为 O(nlgn). 然而在输入的n个元素互异的情况下,利用最大堆,我们可以获得时间复杂度为 O(nlgk)的算法. 1 #include <stdio.h> 2 3 #define N 128 4 5 int heap[N], max_size, cur_pos = 1; 6 7 void adjust(int i) { // bottom up 8 int x = heap[i]; 9 int p = i / 2; 10

【python cookbook】【数据结构与算法】12.找出序列中出现次数最多的元素

问题:找出一个元素序列中出现次数最多的元素是什么 解决方案:collections模块中的Counter类正是为此类问题所设计的.它的一个非常方便的most_common()方法直接告诉你答案. # Determine the most common words in a list words = [ 'look', 'into', 'my', 'eyes', 'look', 'into', 'my', 'eyes', 'the', 'eyes', 'the', 'eyes', 'the', '

log(n)时间内找出数组第i小的数字

参考算法导论9.2 R_Select(int *a,int p,int r,int i){ if(p==r) return a[p]; int q=partition(a,p,r); int k=q-p; if(i==k) return a[q]; else if(i<k) return R_Select(a,p,q-1,i); else return R_Select(a,q+1,r,i-k); } log(n)时间内找出数组第i小的数字