nth_element函数——C++快速选择函数

devtools/2025/2/3 3:41:49/

目录

1. 函数原型

2. 功能描述

3. 算法原理

4. 时间复杂度

5. 空间复杂度

6. 使用示例

8. 注意事项

9. 自定义比较函数

11. 总结


nth_element 是 C++ 标准库中提供的一个算法,位于 <algorithm> 头文件中,用于部分排序序列。它的主要功能是将序列中第 k 小的元素放到第 k 个位置上,并且保证所有在它之前的元素都不大于它,所有在它之后的元素都不小于它。这个算法的核心思想是基于快速排序的分区操作(Quickselect)。

1. 函数原型

nth_element函数原型如下

template <class RandomIt>
void nth_element(RandomIt first, RandomIt nth, RandomIt last);template <class RandomIt, class Compare>
void nth_element(RandomIt first, RandomIt nth, RandomIt last, Compare comp);
  • first:指向序列起始位置的随机访问迭代器。

  • nth:指向序列中第 k 个位置的随机访问迭代器。

  • last:指向序列结束位置的随机访问迭代器。

  • comp(可选):自定义比较函数,默认是 std::less(升序)。

2. 功能描述(无返回值)

(补充:第k大对应第n-k小,对应的下标为n-1-k)

nth_element 的功能是将序列中第 k 小(对应数组下标为k-1)的元素放到第 k 个位置上,并且保证:

  • 所有在第 k 个位置之前的元素都不大于第 k 个位置的元素。

  • 所有在第 k 个位置之后的元素都不小于第 k 个位置的元素。

3. 算法原理

nth_element 的实现基于快速选择算法(Quickselect),它是快速排序算法的一个变种。其主要步骤如下:

  1. 选择支点:从序列中选择一个元素作为支点(pivot)。

  2. 分区操作:将序列分为两部分:

    • 小于等于支点的元素放在支点的左边。

    • 大于支点的元素放在支点的右边。

  3. 递归处理

    • 如果支点的位置恰好是 k,则支点就是第 k 小的元素。

    • 如果支点的位置小于 k,则在右边的子序列中递归处理。

    • 如果支点的位置大于 k,则在左边的子序列中递归处理。

4. 时间复杂度

  • 平均时间复杂度:O(n)。

  • 最坏时间复杂度:O(n2),但这种情况很少发生,可以通过随机选择支点来避免。

5. 空间复杂度

  • 空间复杂度:O(1),不考虑递归调用的栈空间。

6. 使用示例

以下是一个使用 nth_element 的示例代码,用于找到一个数组中第 k 小的元素:

#include <iostream>
#include <vector>
#include <algorithm>int main() {std::vector<int> arr = {7, 10, 4, 3, 20, 15};int k = 3; // 找第 3 小的元素// 使用 nth_elementstd::nth_element(arr.begin(), arr.begin() + k - 1, arr.end());// 输出第 k 小的元素std::cout << "第 " << k << " 小的元素是: " << arr[k - 1] << std::endl;return 0;
}

对于上述代码,输出结果为:

第 3 小的元素是: 7

8. 注意事项

  • nth_element 不会完全排序整个序列,它只保证第 k 小的元素在正确的位置上。

  • 如果需要完全排序,可以使用 std::sort

  • nth_element 的性能优于完全排序(std::sort 的时间复杂度为 O(nlogn)),特别是在只需要找到第 k 小的元素时。

9. 自定义比较函数

如果需要根据自定义规则进行排序,可以使用自定义比较函数。例如,以下代码按降序找到第 k 大的元素:

#include <iostream>
#include <vector>
#include <algorithm>int main() {std::vector<int> arr = {7, 10, 4, 3, 20, 15};int k = 3; // 找第 3 大的元素// 使用 nth_element 和自定义比较函数std::nth_element(arr.begin(), arr.begin() + k - 1, arr.end(), std::greater<int>());// 输出第 k 大的元素std::cout << "第 " << k << " 大的元素是: " << arr[k - 1] << std::endl;return 0;
}

对于上述代码,输出结果为:

第 3 大的元素是: 10

11. 总结

nth_element 是一个非常高效的算法,适用于以下场景:

  • 需要找到序列中第 k 小或第 k 大的元素。

  • 不需要对整个序列进行完全排序。

  • 需要高效地处理大数据量。

收藏加关注,观看不迷路 


http://www.ppmy.cn/devtools/155615.html

相关文章

一觉醒来全球编码能力下降100000倍,新手小白的我决定科普C语言——函数

1. 函数的概念 数学中我们其实就⻅过函数的概念&#xff0c;⽐如&#xff1a;⼀次函数 y kx b &#xff0c;k和b都是常数&#xff0c;给⼀个任意的 x&#xff0c;就得到⼀个y值。其实在C语⾔也引⼊函数&#xff08;function&#xff09;的概念&#xff0c;有些翻译为&#xf…

C# Winform制作一个登录系统

using System; using System.Collections; using System.Collections.Generic; using System.ComponentModel; using System.Data; using System.Drawing; using System.Linq; using System.Text; using System.Threading.Tasks; using System.Windows.Forms;namespace 登录 {p…

android获取EditText内容,TextWatcher按条件触发

android获取EditText内容&#xff0c;TextWatcher按条件触发 背景&#xff1a;解决方案&#xff1a;效果&#xff1a; 背景&#xff1a; 最近在尝试用原生安卓实现仿element-ui表单校验功能&#xff0c;其中涉及到EditText组件内容的动态校验&#xff0c;初步实现功能后&#…

pytorch使用SVM实现文本分类

人工智能例子汇总:AI常见的算法和例子-CSDN博客 完整代码: import torch import torch.nn as nn import torch.optim as optim import jieba import numpy as np from sklearn.model_selection import train_test_split from sklearn.feature_extraction.text import Tfid…

Python | Pytorch | 什么是 Inplace Operation(就地操作)?

如是我闻&#xff1a; 在 PyTorch 中&#xff0c;Inplace Operation&#xff08;就地操作&#xff09;是指直接修改 Tensor 本身&#xff0c;而不是创建新的 Tensor 的操作。PyTorch 中的 Inplace 操作通常会在函数名后加上 _ 作为后缀&#xff0c;例如&#xff1a; tensor.ad…

MongoDB 删除文档

常用的删除文档方法包括 deleteOne()、deleteMany() 以及 findOneAndDelete()。 使用场景&#xff1a; 数据清理&#xff1a;删除不再需要的旧数据或无效数据。数据修正&#xff1a;在数据修正过程中删除错误的或重复的文档。自动化任务&#xff1a;在自动化脚本或任务中&…

Games104——游戏引擎Gameplay玩法系统:基础AI

这里写目录标题 寻路/导航系统NavigationWalkable AreaWaypoint NetworkGridNavigation Mesh&#xff08;寻路网格&#xff09;Sparse Voxel Octree Path FindingDijkstra Algorithm迪杰斯特拉算法A Star&#xff08;A*算法&#xff09; Path Smoothing Steering系统Crowd Simu…

答疑解惑:如何监控EMC unity存储系统磁盘重构rebuild进度

近期有个朋友咨询的问题&#xff0c;这个其实很有代表性的&#xff0c;以前在VNX存储中&#xff0c;通过磁盘的属性是可以看到rebuild的进度的。到了unity年代&#xff0c;更换了一个磁盘&#xff0c;如何查询重构的进度&#xff0c;从图形界面好像没有找到合适的地方去查看。 …