C++之STL整理(4)之set 用法(创建、赋值、增删查改)详解

C++之STL整理(4)之set 用法(创建、赋值、增删查改)详解

注:整理一些突然学到的C++知识,随时mark一下
例如:忘记的关键字用法,新关键字,新数据结构


C++ 的map用法整理

  • C++之STL整理(4)之set 用法(创建、赋值、增删查改)详解
  • 一、set的初始化
    • 1、set初始化
      • (1)构造函数初始化
      • (2)赋值操作
      • (3)swap函数;
    • 3、大小操作
  • 二、set的增删查改
    • 1、插入操作insert
    • 2、删操作erase、clear
    • 3、查找操作 find
    • 4、统计count
  • 三、指定排序规则
    • 1、指定set容器的排序规则
    • 2、 对于自定义数据类型,set 必须指定排序规则
  • 总结


提示:本文为 C++ 中 set构造、赋值、接口 的写法和举例


一、set的初始化

  STL 中的set是一个关联容器,它存储的元素都是键值一体的(key-value pair),并且也会根据键(key)自动排序。set存储的元素(键)是唯一的,不允许键重复,并且默认按照从小到大的顺序进行排序(和map一样也可以通过提供一个自定义的比较函数或对象来改变排序方式,自定义类型必须增加)。std::set 的元素通过其键或迭代器进行访问,并且在这个容器中,键也是值(key is value),也就是说set相当于键值一体的map。

set的基本操作
插入元素:使用 insert 成员函数向 set 中插入元素。如果元素已经存在,则不会插入重复的元素。.insert()的返回值是个pair结构的对组 pair<iterator, bool> bool代表插入是否成功。
查找元素:使用 find 成员函数来查找元素。如果找到元素,则返回指向该元素的迭代器;否则返回 end() 迭代器。
删除元素:使用 erase 成员函数来删除元素。你可以通过迭代器或键来删除元素。
遍历元素:使用迭代器iterator来遍历 set 中的元素。由于 set 是有序的,所以遍历将按照元素的排序顺序进行。

示例
下面是一个简单的 std::set 使用示例:

#include <iostream>  
#include <set>  int main() {  std::set<int> mySet;  // 插入元素  mySet.insert(3);  mySet.insert(1);  mySet.insert(4);  mySet.insert(1); // 不会插入重复的元素  // 遍历并打印元素  for (int num : mySet) {  std::cout << num << " ";  }  std::cout << std::endl; // 输出: 1 3 4  // 查找元素  if (mySet.find(3) != mySet.end()) {  std::cout << "Found 3 in the set." << std::endl;  }  // 删除元素  mySet.erase(4);  // 再次遍历并打印元素  for (int num : mySet) {  std::cout << num << " ";  }  std::cout << std::endl; // 输出: 1 3  return 0;  
}

性能特点
插入和删除操作:由于 std::set 是基于红黑树实现的,所以插入和删除操作的平均时间复杂度为 O(log n),其中 n 是集合中元素的数量。
查找操作:查找操作的平均时间复杂度也是 O(log n)
内存使用:set 通常需要比数组或向量更多的内存,因为它需要存储额外的信息来维护树的平衡。

1、set初始化

(1)构造函数初始化

set<T> st;multiset<T> mst;这两个是默认构造函数,用于创建一个空的 set 或 multiset。

示例:

std::set<int> mySet;  // 创建一个空的整数set  
std::multiset<std::string> myMultiSet;  // 创建一个空的字符串multiset

set(const set &st);multiset(const multiset &mst);
这些是拷贝构造函数,用于创建一个新的 set 或 multiset,它是现有 set 或 multiset 的副本。

示例:

std::set<int> mySet = {1, 2, 3};  
std::set<int> anotherSet(mySet);  // 创建一个mySet的副本

(2)赋值操作

set& operator=(const set &st); 和 multiset& operator=(const multiset &mst);是重载的等号操作符,用于将一个 set 或 multiset 的内容赋值给另一个已经存在的 set 或 multiset。

示例:

std::set<int> mySet = {1, 2, 3};  
std::set<int> anotherSet;  
anotherSet = mySet;  // 将mySet的内容赋给anotherSet

(3)swap函数;

swap 是一个成员函数,用于交换两个 set 或 multiset 的内容。

std::set<int> mySet = {1, 2, 3};  
std::set<int> anotherSet = {4, 5, 6};  
mySet.swap(anotherSet);  // 交换两个set的内容

3、大小操作

成员函数size();返回 set 或 multiset 中元素的数量。
成员函数empty();判断 set 或 multiset 是否为空。如果为空,返回 true;否则返回 false。

示例:

std::set<int> mySet = {1, 2, 3, 4, 5};  
std::cout << "Size of mySet: " << mySet.size() << std::endl;  // 输出: Size of mySet: 5std::set<int> mySet;  
if (mySet.empty()) {  std::cout << "mySet is empty." << std::endl;  
} else {  std::cout << "mySet is not empty." << std::endl;  
}  // 输出: mySet is empty.

二、set的增删查改

1、插入操作insert

insert(elem);用于在 set 或 multiset 中插入一个元素。如果元素已存在(对于 set),则不会重复插入。

std::set<int> mySet;  
mySet.insert(5); // 插入元素5  
mySet.insert(3); // 插入元素3

2、删操作erase、clear

clear();清除 set 或 multiset 中的所有元素。
erase(pos);删掉迭代器 pos 所指向的元素,并返回指向下一个元素的迭代器。

std::set<int> mySet = {1, 2, 3, 4, 5};  
auto it = mySet.find(3); // 查找元素3的迭代器  
if (it != mySet.end()) {  it = mySet.erase(it); // 删除元素3,it现在指向4  
}
mySet.clear(); // 清空set,现在mySet是空的

erase(beg, end);删除区间 [beg, end) 内的所有元素,并返回指向 end 之后元素的迭代器。

std::set<int> mySet = {1, 2, 3, 4, 5};  
auto it = mySet.find(3); // 查找元素3的迭代器  
if (it != mySet.end()) {  mySet.erase(it, std::next(it)); // 删除元素3(包括3)之后的所有元素  
}

erase(elem);删除容器中值为 elem 的元素。对于 set,这只会删除一个元素(如果存在的话),因为 set 中的元素是唯一的。对于 multiset,会删除所有值为 elem 的元素。

std::multiset<int> myMultiSet = {1, 2, 2, 3, 4, 4, 4};  
myMultiSet.erase(2); // 删除一个值为2的元素  
myMultiSet.erase(4); // 删除一个值为4的元素,现在myMultiSet中还有两个4

3、查找操作 find

find(key);在 set 或 multiset 中查找键 key。如果找到,返回指向该键的元素的迭代器;否则返回 end() 迭代器。

std::set<int> mySet = {1, 2, 3, 4, 5};  
auto it = mySet.find(3); // 查找元素3  
if (it != mySet.end()) {  std::cout << "Found: " << *it << std::endl; // 输出: Found: 3  
} else {  std::cout << "Not found." << std::endl;  
}

4、统计count

count(key);对于 set,由于元素是唯一的,count(key) 要么返回 1(如果 key 存在),要么返回 0(如果 key 不存在)。对于 multiset,它返回 key 出现的次数。

std::multiset<int> myMultiSet = {1, 2, 2, 3, 4, 4, 4};  
size_t countOfTwo = myMultiSet.count(2); // 查找元素2出现的次数  
std::cout << "Number of 2s: " << countOfTwo << std::endl; // 输出: Number of 2s: 2

三、指定排序规则

您提到的内容是关于std::set和std::multiset容器的一个重要特性,即可以自定义排序规则。默认情况下,这些容器使用std::less来比较元素,但是可以通过提供自定义的比较函数或仿函数(functor)来改变排序规则。

1、指定set容器的排序规则

创建set或multiset时指定排序规则通常是通过传递一个比较对象(比如函数对象或lambda表达式)给容器的构造函数(第二个参数)来实现的。这个比较对象必须满足比较函数的要求,假设对于任何两个类型T的对象a和b,比较对象comp有以下规则条件:

comp(a, b)在a小于b时返回true
comp(b, a)在b小于a时返回true
comp(a, a)对于任何a都返回false

示例:

#include <iostream>  
#include <set>  
#include <string>  // 自定义比较函数对象  
struct MyCompare {  bool operator()(const std::string& a, const std::string& b) const {  // 这里我们按照字符串的长度进行排序,而不是默认的字典序  return a.size() < b.size();  }  
};  int main() {  // 使用自定义比较对象创建set  std::set<std::string, MyCompare> mySet;  // 插入元素,它们将按照字符串长度排序  mySet.insert("apple");  mySet.insert("banana");  mySet.insert("cherry");  // 输出set中的元素,将看到它们按长度排序  for (const auto& str : mySet) {  std::cout << str << ' ';  }  std::cout << std::endl;  return 0;  
}

2、 对于自定义数据类型,set 必须指定排序规则

当使用自定义数据类型作为set或multiset的键时,通常需要提供一个比较函数或仿函数来定义如何比较这些对象。这是因为std::set和std::multiset需要知道这个新数据类型如何对元素进行排序。

示例:

假设我们有一个自定义的Person类,我们想要根据年龄对Person对象进行排序:

#include <iostream>  
#include <set>  
#include <string>  // 自定义数据类型  
struct Person {  std::string name;  int age;  Person(const std::string& name, int age) : name(name), age(age) {}  
};  // 自定义比较函数对象,用于Person类型的set  
struct PersonCompare {  bool operator()(const Person& a, const Person& b) const {  return a.age < b.age; // 按照年龄升序排序  }  
};  int main() {  // 使用自定义比较对象创建set来存储Person对象  std::set<Person, PersonCompare> peopleSet;  // 插入Person对象  peopleSet.insert(Person("Alice", 30));  peopleSet.insert(Person("Bob", 20));  peopleSet.insert(Person("Charlie", 25));  // 输出set中的Person对象,将看到它们按年龄排序  for (const auto& person : peopleSet) {  std::cout << person.name << ": " << person.age << std::endl;  }  return 0;  
}

在这个例子中,我们定义了一个PersonCompare仿函数来告诉set如何比较Person对象。这样,当我们将Person对象插入到set中时,它们将按照年龄进行排序。

总结

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/778537.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

人才测评系统 提升HR招聘和人岗管理

人才是一个企业的核心竞争力。商业社会的激烈竞争和种种挑战&#xff0c;实际上都是人才的竞争。企业的招聘&#xff0c;职位调动&#xff0c;晋升&#xff0c;人岗匹配的核心对象都是人。现如今&#xff0c;越来越多的企业引入了人才测评机制&#xff0c;但是一些HR却出现不理…

Uibot6.0 (RPA财务机器人师资培训第5天 ) 报销汇总机器人案例实战

训练网站&#xff1a;泓江科技 (lessonplan.cn)https://laiye.lessonplan.cn/list/ec0f5080-e1de-11ee-a1d8-3f479df4d981https://laiye.lessonplan.cn/list/ec0f5080-e1de-11ee-a1d8-3f479df4d981https://laiye.lessonplan.cn/list/ec0f5080-e1de-11ee-a1d8-3f479df4d981https…

华为广告打包报错,问题思考

华为广告打包时报错 fata日志不一样能反映出完整的错误日志信息&#xff0c;仅看fata日志具有误导性&#xff0c;有可能指向错误的方向。 通过看完整的日志可见&#xff0c;错误的原因为 Caused by: java.lang.ClassNotFoundException: com.huawei.hms.ads.base.R$dimenfata日…

SDKMAN多版本SDK并行管理工具

一、简介 SDKMAN是管理多个SDK并行版本的工具&#xff0c;它提供了方便的命令行界面&#xff08;CLI&#xff09;和API&#xff0c;用于列出&#xff0c;安装&#xff0c;切换和删除候选对象。此外&#xff0c;它还为我们设置了环境变量。 它还允许开发人员安装基于JVM的SDK&…

2024/03/28(C++·day4)

一、思维导图 二、练习题 1、写出三种构造函数&#xff0c;算术运算符、关系运算符、逻辑运算符重载尝试实现自增、自减运算符的重载 #include <iostream>using namespace std;// 构造函数示例 class MyClass { private:int data; public:// 默认构造函数MyClass() {da…

vim分屏命令

一、创建分屏 使用如下命令&#xff0c;可以创建多个分屏&#xff0c;如此就可以同时编辑多个文件。 命令英文功能:sp [文件名]split横向增加分屏。:vsp [文件名]vertical split纵向增加分屏。 二、操作分屏 命令英文功能注意wwindow切换到下一个分屏。使用前先按下Ctrl w…

Elasticsearch从入门到精通-07ES底层原理学习

Elasticsearch从入门到精通-07ES底层原理和高级功能 &#x1f44f;作者简介&#xff1a;大家好&#xff0c;我是程序员行走的鱼 &#x1f4d6; 本篇主要介绍和大家一块学习一下ES底层原理包括集群原理、路由原理、分配控制、分配原理、文档分析原理、文档并发安全原理以及一些高…

嵌入式学习44-哈希算法和排序算法

Hash 哈希算法&#xff1a; 在记录的 存储位置 和它的 关键字 之间建立一种去特定的对应关系&#xff0c;使得每个关键字key对应一个存储位置&#xff1b; 查找时&#xff0c;根据确定的对应关系&#xff0c;找到给定的 key 的映射。 记录的存储位置 f&a…

win10 pe制作(官纯,危); git proxy; nvme win7 ; 机械革命极光Z驱动; win7x64 shineLon驱动

win10 pe制作(官纯&#xff0c;危险) 下载 https://learn.microsoft.com/zh-cn/windows-hardware/get-started/adk-install#download-the-adk-for-windows-11 https://go.microsoft.com/fwlink/?linkid2243390 https://go.microsoft.com/fwlink/?linkid2243391 安装 adkw…

Tensorflow2.0笔记 - metrics做损失和准确度信息度量

本笔记主要记录metrics相关的内容&#xff0c;详细内容请参考代码注释&#xff0c;代码本身只使用了Accuracy和Mean。本节的代码基于上篇笔记FashionMnist的代码经过简单修改而来&#xff0c;上篇笔记链接如下&#xff1a; Tensorflow2.0笔记 - FashionMnist数据集训练-CSDN博…

SAP BTP云上一个JVM与DB Connection纠缠的案例

前言 最近在CF (Cloud Foundry) 云平台上遇到一个比较经典的案例。因为牵扯到JVM &#xff08;app进程&#xff09;与数据库连接两大块&#xff0c;稍有不慎&#xff0c;很容易引起不快。 在云环境下&#xff0c;有时候相互扯皮的事蛮多。如果是DB的问题&#xff0c;就会找DB…

DDos系列攻击原理与防御原理

七层防御体系 静态过滤 命中黑名单 对确定是攻击的流量直接加入黑名单&#xff08;源地址命中黑名单直接丢弃&#xff0c;缺乏机动性和扩展性&#xff09; 畸形报文过滤 畸形报文攻击 TCP包含多个标记位&#xff0c;排列组合有规律 • 现象&#xff1a;TCP标记位全为1 …

docker:在ubuntu中运行docker容器

前言 1 本笔记本电脑运行的ubuntu20.04系统 2 docker运行在ubuntu20.04系统 3 docker镜像使用的是ubuntu18.04&#xff0c;这样拉的 docker pull ubuntu:18.04 4 docker容器中运行的是ubuntu18.04的系统&#xff0c;嗯就是严谨 5 这纯粹是学习笔记&#xff0c;实际上没啥价值。…

文件的输入与输出(史上最全)

文件的输入与输出 一个文件是一个存储在磁盘中带有指定名称和目录路径的数据集合。当打开文件进行读写时&#xff0c;它变成一个流。 从根本上说&#xff0c;流是通过通信路径传递的字节序列。有两个主要的流&#xff1a;输入流 和 输出流。输入流用于从文件读取数据&#xf…

指纹浏览器是什么?有哪些好用的推荐?

在网络世界中&#xff0c;保护您的在线隐私和安全非常重要。反检测浏览器是专门为此诞生的工具&#xff0c;旨在通过更改浏览器指纹来帮助您做到这一点&#xff0c;它们使网站、广告商和其他人很难跟踪您的在线行为。 一、什么是反检测浏览器&#xff1f; 您是否想过网站如何检…

C++万物起源:类与对象(二)

一、类的6个默认成员函数 如果一个类中什么成员都没有&#xff0c;简称为空类。 空类中真的什么都没有吗&#xff1f; 并不是&#xff0c;任何类在什么都不写时&#xff0c;编译器会自动生成以下6个默认成员 函数。 默认成员函数&#xff1a;用户没有显式实现&#xff0c;…

动态规划算法及Java实例

动态规划算法的基本概念 动态规划算法是一种解决复杂问题的有效方法&#xff0c;它通过将大问题分解为小问题&#xff0c;然后逐个解决这些小问题&#xff0c;最终通过组合小问题的解来得到大问题的解。这种方法的特点是充分利用了问题的重叠子问题和最优子结构的特性&#xf…

篮球论坛系统的设计与实现|Springboot+ Mysql+Java+ B/S结构(可运行源码+数据库+设计文档)

本项目包含可运行源码数据库LW&#xff0c;文末可获取本项目的所有资料。 推荐阅读100套最新项目持续更新中..... 2024年计算机毕业论文&#xff08;设计&#xff09;学生选题参考合集推荐收藏&#xff08;包含Springboot、jsp、ssmvue等技术项目合集&#xff09; 目录 1. …

Linux根据时间删除文件或目录

《liunx根据时间删除文件》和 《Linux 根据时间删除文件或者目录》已经讲述了根据时间删除文件或目录的方法。 下面我做一些补充&#xff0c;讲述一个具体例子。以删除/home目录下的文件为例。 首先通过命令&#xff1a; ls -l --time-style"%Y-%m-%d %H:%M:%S"…

Redis、Mysql双写情况下,如何保证数据一致

Redis、Mysql双写情况下&#xff0c;如何保证数据一致 场景谈谈数据一致性三个经典的缓存模式Cache-Aside Pattern读流程写流程 Read-Through/Write-Through&#xff08;读写穿透&#xff09;Write behind &#xff08;异步缓存写入&#xff09; 操作缓存的时候&#xff0c;删除…