Find the kth largest element in an unsorted array. Note that it is the kth largest element in the sorted order, not the kth distinct element.

For example,
Given [3,2,1,5,6,4] and k = 2, return 5.

Note:
You may assume k is always valid, 1 ≤ k ≤ array’s length.

Credits:
Special thanks to @mithmatt for adding this problem and creating all test cases.

Hide Tags Divide and Conquer Heap

这道题仅仅想出了使用heap来求解的方法。没有想出怎样使用分治来求解。可是在Discuss上看到了一份关于使用分治法很具体的介绍,而且对于当中的切割部分给出了一个很简洁的代码。


解法一:使用最小堆

这道题既能够使用最小堆也能够使用最大堆来求解。先说使用最小堆的解法。

最小堆中最顶端的元素是堆中最小的,我们构建一个大小为k的最小堆,那么最顶端的元素就是第k大的数。

使用数组中的前k个数构成一个最小堆。那么堆顶元素就是第k大的数,然后从第k+1个数開始遍历数组,假设遍历的数组元素小于堆顶元素,舍弃掉。假设遍历的数组元素大于堆顶元素,将堆顶元素出堆,然后将大于堆顶元素的数组中的元素插入堆中,再次形成堆。这样遍历完数组堆顶元素就是第k大的元素。这样的解法是保证在遍历到数组的不论什么一个元素时堆顶元素都是到遍历到的元素为止的第k大的元素。


runtime:8ms

    int findKthLargest(vector<int>& nums, int k) {
        make_heap(nums.begin(),nums.begin()+k,greater<int>());
        for(int i=k;i<nums.size();i++)
        {
            int top=nums.front();
            if(nums[i]>top)
            {
                pop_heap(nums.begin(),nums.begin()+k,greater<int>());
                nums[k-1]=nums[i];
                push_heap(nums.begin(),nums.begin()+k,greater<int>());
            }
        }
        return nums.front();
    }

解法二:使用最大堆

使用最大堆也能够求解。就是将整个数组中的元素构成一个最大堆,这时堆顶元素是最大的,连续将堆顶元素弹出k-1次后堆顶元素就是第k大的数了。

因为stl中默认构建的是最大堆。所以这样的解法可能会更直观一些。
runtime:8ms

    int findKthLargest(vector<int>& nums, int k) {
        make_heap(nums.begin(), nums.end());
        for (int i=1; i<k; i++){
            pop_heap(nums.begin(), nums.end());
            nums.pop_back();
        }
        return nums[0];
}

解法三:分治法

分治法没有想出来,可是在Discuss中看到了一篇很具体的解答。我将它翻译成了中文。
这样的解法是由高速排序发展起来的。
高速排序中,每一次迭代。我们须要选取一个关键元素pivot,然后将数组切割成三个部分:

  1. 小于关键元素pivot的元素
  2. 等于关键元素pivot的元素
  3. 大于关键元素pivot的元素

如今。以[3,2,1,5,4,6]这个数组为例来分析。

假定每次选取最左端的元素作为关键的元素pivot,这样的情况下,是3,然后我们使用3作为pivot将数组分成上面指定的3个部分,最后结果是[1,2,3,5,4,6]。如今3是第3个元素而且我们知道它也是第3小的元素。

因为上面的切割是将比pivot小的元素放在了pivot的左边,所以pivot当pivot在第k-1位置是是第k小的元素。因为这道题目须要寻找第k大的元素,我们能够改动一下切割过程将比pivot大的元素放在k的左边。

这样。切割完毕后数组变成了[5,6,4,3,1,2],如今3是第4大的元素,假设我们须要寻找第2大的元素。我们知道它是在3左边,假设我们须要第5大的元素,我们知道它是在3右边。

如今简单写出算法的流程:

  1. 初始化left为0。right为nums.size()-1
  2. 切割数组,假设pivot在第k-1位。返回pivot
  3. 假设pivot在k-1右边。更新right为pivot的位置值
  4. 否则更新left为pivot的位置值。

  5. 反复2的步骤

runtime:8ms

     int findKthLargest(vector<int>& nums, int k) {
         int pivot=nums[0];
         int left=0;
         int right=nums.size()-1;
         while(true)
         {
             int pos=partion(nums,left,right);
             if(pos==k-1)   return nums[pos];
             if(pos<k-1)   left=pos+1;
             else right=pos-1;
         }

     }

     int partion(vector<int> &nums,int begin,int end)
     {
         int left=begin+1;
         int right=end;
         while(left<=right)
         {
             if(nums[left]<nums[begin]&&nums[right]>nums[begin])
                swap(nums[left],nums[right]);
             if(nums[left]>=nums[begin]) left++;
             if(nums[right]<=nums[begin]) right--;
         }
         swap(nums[begin],nums[right]);
         return right;
     }