编程基础:详解 C++ 中的 `std::sort` 函数
编程基础:详解 C++ 中的 std::sort 函数
在C++编程中,排序是非常常见的操作,而
std::sort是C++标准库中用于排序的一个高效函数。它提供了灵活的排序功能,可以使用默认排序规则或自定义的比较函数。本文将深入探讨std::sort的用法、参数要求、性能特点以及如何结合实际场景灵活运用。
文章目录
- 编程基础:详解 C++ 中的 `std::sort` 函数
- 摘要
- 一、什么是`std::sort`?
- 1.1 定义
- 1.2 应用场景
- 1.3 基本语法
- 二、`std::sort`的基本用法
- 2.1 默认排序规则示例
- 2.2 自定义排序规则示例
- 三、`std::sort` 的高级用法
- 3.1 对结构体排序
- 3.2 对二维数组排序
- 四、注意事项与常见错误
- 4.1 使用`std::sort`时的注意事项
- 4.2 常见错误
- 五、总结
摘要
本文详细介绍了C++标准库中的std::sort函数。内容包括函数的定义、使用场景、参数要求、基本语法和常见示例。通过具体代码示例,展示了std::sort的基本用法和自定义排序的方式,并结合常见的错误和注意事项,帮助读者掌握如何使用这个强大的排序工具。
一、什么是std::sort?
1.1 定义
std::sort是C++标准库中用于对元素进行排序的函数,位于<algorithm>头文件中。它采用了一种混合排序算法,通常是快速排序和插入排序的结合,能够在大多数情况下提供接近 O(n log n) 的时间复杂度。
核心原理:
std::sort通常基于快速排序实现,但为了优化性能,可能会在较小数据量时切换到插入排序。- 默认情况下,
std::sort按升序排序,但也支持自定义比较函数或仿函数。
1.2 应用场景
- 当需要对容器(如数组、
vector等)中的数据进行排序时,std::sort是高效且易用的工具。 - 适用于需要按特定规则排序的数据结构,如排序成绩、按照字母顺序排列字符串等。
1.3 基本语法
std::sort(Iterator first, Iterator last);
std::sort(Iterator first, Iterator last, Compare comp);
first和last:分别表示排序的范围[first, last),即从first指向的元素开始,到last指向的元素前一个结束。comp:可选的自定义比较函数,定义排序规则。默认按照小于运算符<进行排序。
二、std::sort的基本用法
2.1 默认排序规则示例
以下是一个使用std::sort对整数数组进行默认升序排序的简单示例:
#include <iostream>
#include <algorithm> // 导入 std::sort
#include <vector> // 导入 std::vectorint main() {std::vector<int> numbers = {5, 2, 9, 1, 5, 6}; // 初始化一个整型数组// 使用 std::sort 进行排序std::sort(numbers.begin(), numbers.end());// 输出排序后的结果std::cout << "Sorted numbers: ";for (int num : numbers) {std::cout << num << " ";}std::cout << std::endl;return 0;
}
解释:
- 该示例中,
std::sort(numbers.begin(), numbers.end())按照升序对numbers中的整数排序。 begin()和end()分别返回容器的起始和结束迭代器,指定要排序的范围。
2.2 自定义排序规则示例
假设我们有一组数据,想要按降序排列。这时可以传递一个自定义比较函数或使用lambda表达式。
#include <iostream>
#include <algorithm>
#include <vector>int main() {std::vector<int> numbers = {5, 2, 9, 1, 5, 6}; // 初始化一个整型数组// 使用 std::sort 并传入自定义的比较函数,按降序排序std::sort(numbers.begin(), numbers.end(), [](int a, int b) {return a > b; // 定义降序排序规则:如果 a > b,表示 a 在 b 前});// 输出排序后的结果std::cout << "Sorted numbers in descending order: ";for (int num : numbers) {std::cout << num << " ";}std::cout << std::endl;return 0;
}
解释:
- 这里通过
lambda表达式[](int a, int b) { return a > b; },定义了一个自定义的降序排序规则。 std::sort的第三个参数是自定义的比较函数,它会根据该函数的返回值来确定元素的顺序。
三、std::sort 的高级用法
3.1 对结构体排序
除了简单的数据类型,std::sort还可以用于结构体等复杂数据类型的排序。比如我们有一个包含姓名和年龄的结构体,我们可以根据年龄对结构体进行排序。
#include <iostream>
#include <algorithm>
#include <vector>
#include <string>// 定义 Person 结构体
struct Person {std::string name;int age;
};int main() {std::vector<Person> people = {{"Alice", 30},{"Bob", 25},{"Charlie", 35}};// 根据年龄对 people 进行排序std::sort(people.begin(), people.end(), [](const Person &a, const Person &b) {return a.age < b.age; // 按年龄升序排序});// 输出排序结果std::cout << "Sorted people by age: " << std::endl;for (const auto &person : people) {std::cout << person.name << ": " << person.age << std::endl;}return 0;
}
解释:
- 在该示例中,我们定义了一个
Person结构体,其中包含name和age字段。 - 使用
std::sort对Person对象进行排序时,通过自定义比较函数来指定排序规则(这里按照age升序排序)。
3.2 对二维数组排序
有时,我们可能需要对多维数组(如二维数组或嵌套的vector)进行排序。此时可以根据某一列的值进行排序。
#include <iostream>
#include <algorithm>
#include <vector>int main() {std::vector<std::vector<int>> matrix = {{1, 2, 3},{3, 2, 1},{2, 3, 1}};// 按第二列的值进行排序std::sort(matrix.begin(), matrix.end(), [](const std::vector<int>& a, const std::vector<int>& b) {return a[1] < b[1]; // 按第二列升序排序});// 输出排序后的二维数组std::cout << "Sorted matrix by second column: " << std::endl;for (const auto& row : matrix) {for (int elem : row) {std::cout << elem << " ";}std::cout << std::endl;}return 0;
}
解释:
- 在这个示例中,我们定义了一个二维数组
matrix,并根据第二列的值对整个矩阵进行排序。
四、注意事项与常见错误
4.1 使用std::sort时的注意事项
- 未定义行为:使用
std::sort时,比较函数需要遵循严格弱序规则。即自定义比较函数需要确保满足!comp(b, a)和comp(a, b)两者不可同时为真。 - 大数据量排序:对于非常大的数据集,考虑使用
std::stable_sort,它是稳定排序算法,能够保证相同元素的顺序不变。
4.2 常见错误
-
错误使用比较函数:比较函数必须返回一个布尔值,表示两个元素的相对顺序。如果比较函数不符合这一规则,可能会导致未定义行为。
错误示例:
std::sort(vec.begin(), vec.end(), [](int a, int b) {return a - b; // 错误!返回的是整数而非布尔值 });
五、总结
std::sort是C++中功能强大的排序函数,能够在大多数场景下高效排序数据。它可以用于简单的升序或降序排序,也可以通过自定义比较函数实现复杂的排序逻辑。无论是处理简单的数据类型,还是自定义的复杂结构,std::sort都能够提供灵活的排序支持。
| 特性 | 描述 |
|---|---|
| 默认排序 | 使用<运算符,升序排序 |
| 自定义排序 | 通过传入比较函数,自定义排序规则 |
| 稳定性 | std::sort不保证稳定性 |
| 复杂度 | 最佳为 O(n log n),最差为 O(n^2) |
✨ 我是专业牛,一个渴望成为大牛🏆的985硕士🎓,热衷于分享知识📚,帮助他人解决问题💡,为大家提供科研、竞赛等方面的建议和指导🎯。无论是科研项目🛠️、竞赛🏅,还是图像🖼️、通信📡、计算机💻领域的论文辅导📑,我都以诚信为本🛡️,质量为先!🤝
如果你觉得这篇文章对你有所帮助,别忘了点赞👍、收藏📌和关注🔔!你的支持是我继续分享知识的动力🚀!✨ 如果你有任何问题或需要帮助,随时留言📬或私信📲,我都会乐意解答!😊
相关文章:
编程基础:详解 C++ 中的 `std::sort` 函数
编程基础:详解 C 中的 std::sort 函数 在C编程中,排序是非常常见的操作,而std::sort是C标准库中用于排序的一个高效函数。它提供了灵活的排序功能,可以使用默认排序规则或自定义的比较函数。本文将深入探讨std::sort的用法、参数要…...
51单片机的宠物自动投喂系统【proteus仿真+程序+报告+原理图+演示视频】
1、主要功能 该系统由AT89C51/STC89C52单片机LCD1602显示模块温湿度传感器DS1302时钟模块蓝牙步进电机按键、蜂鸣器等模块构成。适用于猫猫/狗狗宠物自动喂食器等相似项目。 可实现基本功能: 1、LCD1602实时显示北京时间和温湿度 2、温湿度传感器DHT11采集环境温湿度 3、时…...
MongoDB快速实战与基本原理
目录 链接:https://note.youdao.com/ynoteshare/index.html?id=5e038498891617c552667b853742fdc1&type=note&_time=1727935558812 Mongo数据库的特点: mongo数据库和关系型数据库的区别: 编辑 关系型数据库和文档型数据库的主要概念对比: 下载和启动(具体…...
编程技巧:优化
第一种:构造回文串---构造法 题目描述 [NOIP2016 普及组] 回文日期 - 洛谷 那么这道题我们总结一些题目要求: 1.输入两个字符串,为起始和终止日期 2.年份不会出现前导0 3.如果是回文日期,答案1 4.如果月份是2,要…...
pycharm中使用anaconda创建多环境,无法将“pip”项识别为 cmdlet、函数、脚本文件或可运行程序的名称
问题描述 用的IDE是: 使用anaconda创建了一个Python 3.9的环境 结果使用pip命令的时候,报错 无法将“pip”项识别为 cmdlet、函数、脚本文件或可运行程序的名称 解决方案 为了不再增加系统变量,我们直接将变量添加在当前项目中你的Ter…...
【Linux】进程周边之优先级
目录 一、优先级 1.为什么要有进程优先级? 2.什么是进程优先级? 3.优先级的初始设定 3.1 PRI 和 NI 3.2如何修改优先级?(sudo/root) 3.2.1 概念: 3.2.2 如何查看进程的优先级? 3.3.3 或…...
Linux高级编程_29_信号
文章目录 进程间通讯 - 信号信号完整的信号周期信号的编号信号的产生发送信号1 kill 函数(他杀)作用:语法:示例: 2 raise函数(自杀)作用:示例: 3 abort函数(自杀)作用:语法:示例: 4 …...
uniapp修改uni-ui组件样式(对微信小程序/H5有效,vue3)
寻找要修改的样式 使用开发者工具找到具体要修改的class类名 修改 <style lang"scss">//.nav为上一级的class.nav::v-deep .uni-navbar--border {border-bottom-style: none !important;} </style>完整代码 <template><view><uni-na…...
Python 封装 socket 为 [TCP/UDP/MULTICAST] 服务端
在新线程中创建 TCP/UDP/MULTICAST 协议的服务端套接字,接收客户端的连接请求或数据,并调用 on_recv 回调函数处理数据。 #!/usr/bin/env python # -*- coding: utf-8 -*- import socket import threading import multiprocessingclass ServerSocket:de…...
c++ STL库 unordered_map
#include <iostream #include <string #include <unordered_map int main() { // 创建一个 unordered_map std::unordered_map<std::string, int> wordCount; // 插入元素 wordCount["apple"] 1; wordCount["banana"] 2;// 使用 insert…...
【接口测试】任务1:登录接口
需要技能竞赛软件测试资料的同学们可s聊我,详细了解 任务实现要求 根据系统管理员—登录—接口API文档,编写接口测试用例,分别使用PostMan及JMeter进行接口测试,需要检查系统接口是否能正常工作,返回值是否正确&#…...
二、Spring Boot集成Spring Security之实现原理
Spring Boot集成Spring Security之实现原理 一、Spring Security实现原理概要介绍二、使用WebSecurityConfiguration向Spring容器中注册FilterChainProxy类型的对象springSecurityFilterChain1、未配置securityFilterChain过滤器链时使用默认配置用于生成默认securityFilterCha…...
基于深度学习的点云处理模型PointNet++学习记录
前面我们已经学习了Open3D,并掌握了其相关应用,但我们也发现对于一些点云分割任务,我们采用聚类等方法的效果似乎并不理想,这时,我们可以想到在深度学习领域是否有相关的算法呢,今天,我们便来学…...
Javascript Object.assgin()详解以及深浅拷贝
Object.assign() 方法是 JavaScript 中用于将所有可枚举属性的值从一个或多个源对象复制到目标对象的方法。它将返回目标对象。这是一种浅拷贝,也就是说,如果源对象中的属性是一个对象或数组,那么这个属性的引用将被复制,而不是对…...
Redis篇(应用案例 - UV统计)(持续更新迭代)
目录 一、HyperLogLog 二、测试百万数据的统计 一、HyperLogLog 首先我们搞懂两个概念: UV:全称Unique Visitor,也叫独立访客量,是指通过互联网访问、浏览这个网页的自然人。 1天内同一个用户多次访问该网站,只记录…...
解锁微信小程序新技能:ECharts动态折线图搭配WebSocket,数据刷新快人一步!
在微信小程序中,数据可视化展示越来越受到开发者的重视。本文将为您介绍如何在微信小程序中使用ECharts绘制折线图,并通过WebSocket实现实时更新图表数据。 一、准备工作 创建微信小程序项目 首先,我们需要创建一个微信小程序项目。如果您已…...
上交所服务器崩溃:金融交易背后的技术隐患暴露杭州BGP高防服务器43.228.71.X
一、上交所宕机事件始末 2024 年 9 月 27 日,上交所交易系统突发崩溃,这一事件犹如一颗巨石投入平静的湖面,引起了轩然大波。当天上午,众多投资者反馈券商交易出现延迟问题,随后上交所发布了《关于股票竞价交易出现异常…...
P4、P4D、HelixSwarm 各种技术问题咨询
多年大型项目P4仓库运维经验,为你解决各种部署以及标准工业化流程问题。 Perforce 官网SDPHelixCore GuideHelixSwarm GuideHelixSwarm Download...
Linux 应用层协议HTTP
文章目录 一、初始HTTP协议二、URL格式网络中怎么通过URL进行定位资源呢?编码和解码 三、HTTP的请求格式和响应格式HTTP的请求格式HTTP的响应格式HTTP的请求方法GET方法POST方法GET Vs PostHTTP的封装和分用文件流操作浏览器获得一个完整的网页流程 HTTP的状态码对3…...
Python和C++混淆矩阵地理学医学物理学视觉语言模型和算法模型评估工具
🎯要点 优化损失函数评估指标海岸线检测算法评估遥感视觉表征和文本增强乳腺癌预测模型算法液体中闪烁光和切伦科夫光分离多标签分类任务性能评估有向无环图、多路径标记和非强制叶节点预测二元分类评估特征归因可信性评估马修斯相关系数对比其他准确度 Python桑…...
React Native 开发环境搭建(全平台详解)
React Native 开发环境搭建(全平台详解) 在开始使用 React Native 开发移动应用之前,正确设置开发环境是至关重要的一步。本文将为你提供一份全面的指南,涵盖 macOS 和 Windows 平台的配置步骤,如何在 Android 和 iOS…...
阿里云ACP云计算备考笔记 (5)——弹性伸缩
目录 第一章 概述 第二章 弹性伸缩简介 1、弹性伸缩 2、垂直伸缩 3、优势 4、应用场景 ① 无规律的业务量波动 ② 有规律的业务量波动 ③ 无明显业务量波动 ④ 混合型业务 ⑤ 消息通知 ⑥ 生命周期挂钩 ⑦ 自定义方式 ⑧ 滚的升级 5、使用限制 第三章 主要定义 …...
Linux简单的操作
ls ls 查看当前目录 ll 查看详细内容 ls -a 查看所有的内容 ls --help 查看方法文档 pwd pwd 查看当前路径 cd cd 转路径 cd .. 转上一级路径 cd 名 转换路径 …...
ServerTrust 并非唯一
NSURLAuthenticationMethodServerTrust 只是 authenticationMethod 的冰山一角 要理解 NSURLAuthenticationMethodServerTrust, 首先要明白它只是 authenticationMethod 的选项之一, 并非唯一 1 先厘清概念 点说明authenticationMethodURLAuthenticationChallenge.protectionS…...
【Web 进阶篇】优雅的接口设计:统一响应、全局异常处理与参数校验
系列回顾: 在上一篇中,我们成功地为应用集成了数据库,并使用 Spring Data JPA 实现了基本的 CRUD API。我们的应用现在能“记忆”数据了!但是,如果你仔细审视那些 API,会发现它们还很“粗糙”:有…...
PL0语法,分析器实现!
简介 PL/0 是一种简单的编程语言,通常用于教学编译原理。它的语法结构清晰,功能包括常量定义、变量声明、过程(子程序)定义以及基本的控制结构(如条件语句和循环语句)。 PL/0 语法规范 PL/0 是一种教学用的小型编程语言,由 Niklaus Wirth 设计,用于展示编译原理的核…...
[Java恶补day16] 238.除自身以外数组的乘积
给你一个整数数组 nums,返回 数组 answer ,其中 answer[i] 等于 nums 中除 nums[i] 之外其余各元素的乘积 。 题目数据 保证 数组 nums之中任意元素的全部前缀元素和后缀的乘积都在 32 位 整数范围内。 请 不要使用除法,且在 O(n) 时间复杂度…...
【开发技术】.Net使用FFmpeg视频特定帧上绘制内容
目录 一、目的 二、解决方案 2.1 什么是FFmpeg 2.2 FFmpeg主要功能 2.3 使用Xabe.FFmpeg调用FFmpeg功能 2.4 使用 FFmpeg 的 drawbox 滤镜来绘制 ROI 三、总结 一、目的 当前市场上有很多目标检测智能识别的相关算法,当前调用一个医疗行业的AI识别算法后返回…...
大数据学习(132)-HIve数据分析
🍋🍋大数据学习🍋🍋 🔥系列专栏: 👑哲学语录: 用力所能及,改变世界。 💖如果觉得博主的文章还不错的话,请点赞👍收藏⭐️留言Ǵ…...
作为测试我们应该关注redis哪些方面
1、功能测试 数据结构操作:验证字符串、列表、哈希、集合和有序的基本操作是否正确 持久化:测试aof和aof持久化机制,确保数据在开启后正确恢复。 事务:检查事务的原子性和回滚机制。 发布订阅:确保消息正确传递。 2、性…...
