黑马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++ HTTP框架推荐(特点及优势)

《C++HTTP框架推荐(特点及优势)》:本文主要介绍C++HTTP框架推荐的相关资料,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录1. Crow2. Drogon3. Pistache4. cpp-httplib5. Beast (Boos

C++类和对象之初始化列表的使用方式

《C++类和对象之初始化列表的使用方式》:本文主要介绍C++类和对象之初始化列表的使用方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录C++初始化列表详解:性能优化与正确实践什么是初始化列表?初始化列表的三大核心作用1. 性能优化:避免不必要的赋值操作2. 强

C++迭代器失效的避坑指南

《C++迭代器失效的避坑指南》在C++中,迭代器(iterator)是一种类似指针的对象,用于遍历STL容器(如vector、list、map等),迭代器失效是指在对容器进行某些操作后... 目录1. 什么是迭代器失效?2. 哪些操作会导致迭代器失效?2.1 vector 的插入操作(push_back,

SpringIOC容器Bean初始化和销毁回调方式

《SpringIOC容器Bean初始化和销毁回调方式》:本文主要介绍SpringIOC容器Bean初始化和销毁回调方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐... 目录前言1.@Bean指定初始化和销毁方法2.实现接口3.使用jsR250总结前言Spring Bea

C#如何调用C++库

《C#如何调用C++库》:本文主要介绍C#如何调用C++库方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录方法一:使用P/Invoke1. 导出C++函数2. 定义P/Invoke签名3. 调用C++函数方法二:使用C++/CLI作为桥接1. 创建C++/CL

C++如何通过Qt反射机制实现数据类序列化

《C++如何通过Qt反射机制实现数据类序列化》在C++工程中经常需要使用数据类,并对数据类进行存储、打印、调试等操作,所以本文就来聊聊C++如何通过Qt反射机制实现数据类序列化吧... 目录设计预期设计思路代码实现使用方法在 C++ 工程中经常需要使用数据类,并对数据类进行存储、打印、调试等操作。由于数据类

Linux下如何使用C++获取硬件信息

《Linux下如何使用C++获取硬件信息》这篇文章主要为大家详细介绍了如何使用C++实现获取CPU,主板,磁盘,BIOS信息等硬件信息,文中的示例代码讲解详细,感兴趣的小伙伴可以了解下... 目录方法获取CPU信息:读取"/proc/cpuinfo"文件获取磁盘信息:读取"/proc/diskstats"文

一文详解如何在Python中从字符串中提取部分内容

《一文详解如何在Python中从字符串中提取部分内容》:本文主要介绍如何在Python中从字符串中提取部分内容的相关资料,包括使用正则表达式、Pyparsing库、AST(抽象语法树)、字符串操作... 目录前言解决方案方法一:使用正则表达式方法二:使用 Pyparsing方法三:使用 AST方法四:使用字

C++使用printf语句实现进制转换的示例代码

《C++使用printf语句实现进制转换的示例代码》在C语言中,printf函数可以直接实现部分进制转换功能,通过格式说明符(formatspecifier)快速输出不同进制的数值,下面给大家分享C+... 目录一、printf 原生支持的进制转换1. 十进制、八进制、十六进制转换2. 显示进制前缀3. 指

C++中初始化二维数组的几种常见方法

《C++中初始化二维数组的几种常见方法》本文详细介绍了在C++中初始化二维数组的不同方式,包括静态初始化、循环、全部为零、部分初始化、std::array和std::vector,以及std::vec... 目录1. 静态初始化2. 使用循环初始化3. 全部初始化为零4. 部分初始化5. 使用 std::a