leetcode 数组中第k个最大元素

给定一个整数数组,返回数组的第k个最大的元素。

(1)很容易能想到,可以先对数组进行排序,排序之后,nums[size - k ]就是想要的结果。排序算法可以使用选择排序、交换排序、插入排序、堆排序、快速排序、归并排序。前3种排序算法的时间复杂度为O(n * n),后3种排序算法的事件复杂度是O(nlogn)。在实际使用中,常常使用堆排序或者快速排序,堆排序是选择排序的思想,快速排序是交换排序的思想。

(2)使用堆排序或者快速排序,需要把数组整体排序完成吗?不需要

因为题目要求查找第k个最大的数,如果使用堆排序,那么进行k次选择就可以得到结果。如果使用快速排序,快速排序使用了二分法和递归的思想,每次运算都能确定二分的这个点的位置,如果这个点的位置正好是size - k的位置,那么这个位置就是题目的结果,此时就可以停止排序。

1堆排序

堆排序算法

cpp 复制代码
class Solution {
public:
    int findKthLargest(vector<int>& nums, int k) {
      createInitialHeap(nums);
      for (int i = 0; i < k; i++) {
        int tmp = nums[0];
        nums[0] = nums[nums.size() - i - 1];
        nums[nums.size() - i - 1] = tmp;
        adjustHeap(nums, 0, nums.size() - i - 1 - 1);
      }
      return nums[nums.size() - k];
    }

    void createInitialHeap(vector<int>& nums) {
      int mid = nums.size() / 2;
      for (int i = mid; i >= 0; i--) {
        adjustHeap(nums, i, nums.size() - 1);
      }
    }

    void adjustHeap(vector<int>& nums, int start_index, int end_index) {
      if (start_index >= end_index) {
        return;
      }

      int bigger_index = start_index;
      int left_child_index = start_index * 2 + 1;
      int right_child_index = left_child_index + 1;

      if (left_child_index <= end_index && nums[left_child_index] > nums[bigger_index]) {
        bigger_index = left_child_index;
      }

      if (right_child_index <= end_index && nums[right_child_index] > nums[bigger_index]) {
        bigger_index = right_child_index;
      }

      if (bigger_index != start_index) {
        int tmp = nums[start_index];
        nums[start_index] = nums[bigger_index];
        nums[bigger_index] = tmp;
        adjustHeap(nums, bigger_index, end_index);
      }
    }
};

2快速排序

cpp 复制代码
class Solution {
public:
    int findKthLargest(vector<int>& nums, int k) {
      quickSort(nums, 0, nums.size() - 1, k);
      return nums[nums.size() - k];
    }

    void quickSort(vector<int> &nums, int left, int right, int k) {
        if (left >= right) {
            return;
        }

        int select_value = nums[left];
        int tmp_left = left;
        int tmp_right = right;
        while (tmp_left < tmp_right) {
            while (tmp_right > tmp_left && nums[tmp_right] > select_value) {
                tmp_right--;
            }

            if (tmp_right > tmp_left) {
                nums[tmp_left] = nums[tmp_right];
                tmp_left++;
            }

            while (tmp_left < tmp_right && nums[tmp_left] < select_value) {
                tmp_left++;
            }

            if (tmp_left < tmp_right) {
                nums[tmp_right] = nums[tmp_left];
                tmp_right--;
            }
        }
        nums[tmp_left] = select_value;

        if (tmp_left == nums.size() - k) {
            return;
        } else if (tmp_left < nums.size() - k) {
            quickSort(nums, tmp_left + 1, right, k);
        } else {
            quickSort(nums, left, tmp_left - 1, k);
        }
    }
};
相关推荐
繁依Fanyi12 分钟前
简易安卓句分器实现
java·服务器·开发语言·算法·eclipse
烦躁的大鼻嘎28 分钟前
模拟算法实例讲解:从理论到实践的编程之旅
数据结构·c++·算法·leetcode
C++忠实粉丝1 小时前
计算机网络socket编程(4)_TCP socket API 详解
网络·数据结构·c++·网络协议·tcp/ip·计算机网络·算法
祁思妙想1 小时前
10.《滑动窗口篇》---②长度最小的子数组(中等)
leetcode·哈希算法
用户37791362947551 小时前
【循环神经网络】只会Python,也能让AI写出周杰伦风格的歌词
人工智能·算法
福大大架构师每日一题1 小时前
文心一言 VS 讯飞星火 VS chatgpt (396)-- 算法导论25.2 1题
算法·文心一言
EterNity_TiMe_1 小时前
【论文复现】(CLIP)文本也能和图像配对
python·学习·算法·性能优化·数据分析·clip
机器学习之心2 小时前
一区北方苍鹰算法优化+创新改进Transformer!NGO-Transformer-LSTM多变量回归预测
算法·lstm·transformer·北方苍鹰算法优化·多变量回归预测·ngo-transformer
yyt_cdeyyds2 小时前
FIFO和LRU算法实现操作系统中主存管理
算法
alphaTao2 小时前
LeetCode 每日一题 2024/11/18-2024/11/24
算法·leetcode