黑马c++ STL部分 笔记(8) set/ multiset 容器

2024-03-02 19:28

本文主要是介绍黑马c++ STL部分 笔记(8) set/ multiset 容器,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

简介
所有元素都会在插入时自动被排序
本质
set/multiset属于关联式容器,底层结构是用二叉树实现。
set和multiset区别
set允许容器中有重复的元素
multiset允许容器中有重复的元素

 1.set构造和赋值

// set构造和赋值
/*
构造:
set<T> st; //默认构造函数:
set(const set &st); //拷贝构造函数
赋值:
set& operator=(const set &st); //重载等号操作符
*/
#include <bits/stdc++.h>
using namespace std;
void printset(set<int> &s)
{for (set<int>::iterator it = s.begin(); it != s.end(); it++){cout << *it << " ";}cout << endl;
}void test01()
{set<int> s1;s1.insert(10); // 插入数据只有insert方式s1.insert(30);s1.insert(40);s1.insert(20);s1.insert(30);// set里的元素自动被排序,不允许插入重复的值printset(s1); // 10 20 30 40// 拷贝构造set<int> s2(s1);printset(s2); // 10 20 30 40// 赋值set<int> s3;s3 = s2;printset(s3); // 10 20 30 40
}
int main()
{test01();
}
/*
总结:
set容器插入数据时用insert
set容器插入数据的数据会自动排序
*/

2. set大小和交换

// set大小和交换
/*不支持resize
size(); //返回容器中元素的数目
empty(); //判断容器是否为空
swap(st); //交换两个集合容器
*/
#include <bits/stdc++.h>
using namespace std;
void printset(set<int> &s)
{for (set<int>::iterator it = s.begin(); it != s.end(); it++){cout << *it << " ";}cout << endl;
}void test01()
{set<int> s1;s1.insert(10);s1.insert(30);s1.insert(40);s1.insert(20);s1.insert(30);printset(s1); // 10 20 30 40if (s1.empty()){cout << "s1为空" << endl;}else{cout << "s1为不空" << endl;cout << s1.size() << endl; // 4}set<int> s2;s2.insert(10);printset(s2); // 10s1.swap(s2);printset(s1); // 10printset(s2); // 10 20 30 40
}
int main()
{test01();
}
/*
总结:
统计大小 — size
判断是否为空 — empty
交换容器 — swap
*/

 3.set插入和删除

// set插入和删除
/*
insert(elem); //在容器中插入元素。
clear(); //清除所有元素
erase(pos); //删除pos'迭代器'所指的元素,返回下一个元素的迭代器。
erase(beg, end); //删除区间[beg,end)的所有元素 ,返回下一个元素的迭代器。
erase(elem); //删除容器中'值'为elem的元素。(类似remove)
*/
#include <bits/stdc++.h>
using namespace std;
void printset(set<int> &s)
{for (set<int>::iterator it = s.begin(); it != s.end(); it++){cout << *it << " ";}cout << endl;
}void test01()
{set<int> s1;// 插入s1.insert(30);s1.insert(10);s1.insert(40);s1.insert(20);s1.insert(30);printset(s1); // 10 20 30 40// 删除迭代器位置s1.erase(s1.begin());printset(s1); // 20 30 40// 删除元素s1.erase(40);printset(s1); // 20 30// 清空s1.clear();printset(s1);
}
int main()
{test01();
}
/*
总结:
插入 — insert
删除 — erase
清空 — clear
*/

4.set查找和统计 

// set查找和统计
/*
find(key); //查找key是否存在,若存在,返回该键的元素的'迭代器';若不存在,返回'set.end()';
count(key); //统计key的'元素'个数  set中返回0/1 multiset为0/>0的值
*/
#include <bits/stdc++.h>
using namespace std;
void printset(set<int> &s)
{for (set<int>::iterator it = s.begin(); it != s.end(); it++){cout << *it << " ";}cout << endl;
}void test01()
{set<int> s1;// 插入s1.insert(30);s1.insert(10);s1.insert(40);s1.insert(20);s1.insert(30);printset(s1); // 10 20 30 40// 查找set<int>::iterator pos = s1.find(40);if (pos != s1.end()){cout << "找到元素:" << *pos << endl; // 找到元素:40}else{cout << "未找到元素:" << endl;}// 统计int num = s1.count(30); // 统计30的个数cout << num << endl;    // 1
}
int main()
{test01();
}
/*
总结:
查找 — find (返回的是迭代器)
统计 — count (对于set,结果为0或者1)
*/

5.set和multiset区别 

// set和multiset区别
/*
区别:
set不可以插入重复数据,而multiset可以
set插入数据的同时会返回插入结果,表示插入是否成功
multiset不会检测数据,因此可以插入重复数据
*/
#include <bits/stdc++.h>
using namespace std;
void test01()
{set<int> s;// set插入pair<set<int>::iterator, bool> ret = s.insert(10);if (ret.second){cout << "第一次插入成功" << endl; // √}else{cout << "第一次插入失败" << endl; // ×}// set重复插入ret = s.insert(10);if (ret.second){cout << "第二次插入成功" << endl; // ×}else{cout << "第二次插入失败" << endl; // √}multiset<int> s2;// multiset插入s2.insert(10);// multiset重复插入s2.insert(10);for (multiset<int>::iterator it = s2.begin(); it != s2.end(); it++){cout << *it << " "; // 10 10}
}
int main()
{test01();
}
/*
总结:
如果不允许插入重复数据可以利用set
如果需要插入重复数据利用multiset
*/

6. pair对组创建

// pair对组创建
/*
两种创建方式:
pair<type, type> p ( value1, value2 );
pair<type, type> p = make_pair( value1, value2 );
*/
#include <bits/stdc++.h>
using namespace std;
void test01()
{pair<string, int> p("Tom", 20);cout << p.first << " " << p.second << endl; // Tom 20pair<string, int> p2 = make_pair("Jerry", 15);cout << p2.first << " " << p2.second << endl; // Jerry 15
}
int main()
{test01();
}
/*
总结:
两种方式都可以创建对组,记住一种即可
*/

7.1  set容器排序
示例一 set存放内置数据类型 

//  set容器排序
/*
学习目标:
set容器默认排序规则为从小到大,掌握如何改变排序规则
主要技术点:
利用仿函数,可以改变排序规则
*/
// 示例一 set存放内置数据类型
#include <bits/stdc++.h>
using namespace std;
class cmp // 仿函数(类)
{
public:bool operator()(int v1, int v2){return v1 > v2;}
};
void test01()
{// 升序set<int> s1;s1.insert(10);s1.insert(30);s1.insert(20);s1.insert(50);s1.insert(40);for (set<int>::iterator it = s1.begin(); it != s1.end(); it++){cout << *it << " "; // 10 20 30 40 50}cout << endl;// 降序,在创建容器时就指定排序规则set<int, cmp> s2;s2.insert(10);s2.insert(30);s2.insert(20);s2.insert(50);s2.insert(40);for (set<int, cmp>::iterator it = s2.begin(); it != s2.end(); it++){cout << *it << " "; // 50 40 30 20 10}cout << endl;
}
int main()
{test01();
}
/*
总结:
对于自定义数据类型,set必须指定排序规则才可以插入数据
*/

7.2  set容器排序
示例二 set存放自定义数据类型 

//  set容器排序
// 示例二 set存放自定义数据类型
#include <bits/stdc++.h>
using namespace std;class person
{
public:person(string name, int age){this->name = name;this->age = age;}string name;int age;
};
class cmp
{
public:bool operator()(person p1, person p2){// 按年龄降序return p1.age > p2.age;}
};
void test01()
{/// 自定义的类型 都会指定排序类型set<person, cmp> s;person s1("刘备", 24);person s2("关羽", 28);person s3("张飞", 25);person s4("赵云", 21);s.insert(s1);s.insert(s2);s.insert(s3);s.insert(s4);for (set<person, cmp>::iterator it = s.begin(); it != s.end(); it++){cout << (*it).name << " " << (*it).age;/*关羽 28张飞 25刘备 24赵云 21*/cout << endl;}
}
int main()
{test01();
}
/*
总结:
对于自定义数据类型,set必须指定排序规则才可以插入数据
*/

这篇关于黑马c++ STL部分 笔记(8) set/ multiset 容器的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/767194

相关文章

C++中unordered_set哈希集合的实现

《C++中unordered_set哈希集合的实现》std::unordered_set是C++标准库中的无序关联容器,基于哈希表实现,具有元素唯一性和无序性特点,本文就来详细的介绍一下unorder... 目录一、概述二、头文件与命名空间三、常用方法与示例1. 构造与析构2. 迭代器与遍历3. 容量相关4

C++中悬垂引用(Dangling Reference) 的实现

《C++中悬垂引用(DanglingReference)的实现》C++中的悬垂引用指引用绑定的对象被销毁后引用仍存在的情况,会导致访问无效内存,下面就来详细的介绍一下产生的原因以及如何避免,感兴趣... 目录悬垂引用的产生原因1. 引用绑定到局部变量,变量超出作用域后销毁2. 引用绑定到动态分配的对象,对象

Redis中Set结构使用过程与原理说明

《Redis中Set结构使用过程与原理说明》本文解析了RedisSet数据结构,涵盖其基本操作(如添加、查找)、集合运算(交并差)、底层实现(intset与hashtable自动切换机制)、典型应用场... 目录开篇:从购物车到Redis Set一、Redis Set的基本操作1.1 编程常用命令1.2 集

Java JUC并发集合详解之线程安全容器完全攻略

《JavaJUC并发集合详解之线程安全容器完全攻略》Java通过java.util.concurrent(JUC)包提供了一整套线程安全的并发容器,它们不仅是简单的同步包装,更是基于精妙并发算法构建... 目录一、为什么需要JUC并发集合?二、核心并发集合分类与详解三、选型指南:如何选择合适的并发容器?在多

C++读写word文档(.docx)DuckX库的使用详解

《C++读写word文档(.docx)DuckX库的使用详解》DuckX是C++库,用于创建/编辑.docx文件,支持读取文档、添加段落/片段、编辑表格,解决中文乱码需更改编码方案,进阶功能含文本替换... 目录一、基本用法1. 读取文档3. 添加段落4. 添加片段3. 编辑表格二、进阶用法1. 文本替换2

C++中处理文本数据char与string的终极对比指南

《C++中处理文本数据char与string的终极对比指南》在C++编程中char和string是两种用于处理字符数据的类型,但它们在使用方式和功能上有显著的不同,:本文主要介绍C++中处理文本数... 目录1. 基本定义与本质2. 内存管理3. 操作与功能4. 性能特点5. 使用场景6. 相互转换核心区别

python语言中的常用容器(集合)示例详解

《python语言中的常用容器(集合)示例详解》Python集合是一种无序且不重复的数据容器,它可以存储任意类型的对象,包括数字、字符串、元组等,下面:本文主要介绍python语言中常用容器(集合... 目录1.核心内置容器1. 列表2. 元组3. 集合4. 冻结集合5. 字典2.collections模块

Spring Boot中获取IOC容器的多种方式

《SpringBoot中获取IOC容器的多种方式》本文主要介绍了SpringBoot中获取IOC容器的多种方式,包括直接注入、实现ApplicationContextAware接口、通过Spring... 目录1. 直接注入ApplicationContext2. 实现ApplicationContextA

linux配置podman阿里云容器镜像加速器详解

《linux配置podman阿里云容器镜像加速器详解》本文指导如何配置Podman使用阿里云容器镜像加速器:登录阿里云获取专属加速地址,修改Podman配置文件并移除https://前缀,最后拉取镜像... 目录1.下载podman2.获取阿里云个人容器镜像加速器地址3.更改podman配置文件4.使用po

k8s容器放开锁内存限制问题

《k8s容器放开锁内存限制问题》nccl-test容器运行mpirun时因NCCL_BUFFSIZE过大导致OOM,需通过修改docker服务配置文件,将LimitMEMLOCK设为infinity并... 目录问题问题确认放开容器max locked memory限制总结参考:https://Access