黑马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++统计函数执行时间的最佳实践

《C++统计函数执行时间的最佳实践》在软件开发过程中,性能分析是优化程序的重要环节,了解函数的执行时间分布对于识别性能瓶颈至关重要,本文将分享一个C++函数执行时间统计工具,希望对大家有所帮助... 目录前言工具特性核心设计1. 数据结构设计2. 单例模式管理器3. RAII自动计时使用方法基本用法高级用法

深入解析C++ 中std::map内存管理

《深入解析C++中std::map内存管理》文章详解C++std::map内存管理,指出clear()仅删除元素可能不释放底层内存,建议用swap()与空map交换以彻底释放,针对指针类型需手动de... 目录1️、基本清空std::map2️、使用 swap 彻底释放内存3️、map 中存储指针类型的对象

C++ STL-string类底层实现过程

《C++STL-string类底层实现过程》本文实现了一个简易的string类,涵盖动态数组存储、深拷贝机制、迭代器支持、容量调整、字符串修改、运算符重载等功能,模拟标准string核心特性,重点强... 目录实现框架一、默认成员函数1.默认构造函数2.构造函数3.拷贝构造函数(重点)4.赋值运算符重载函数

C++ vector越界问题的完整解决方案

《C++vector越界问题的完整解决方案》在C++开发中,std::vector作为最常用的动态数组容器,其便捷性与性能优势使其成为处理可变长度数据的首选,然而,数组越界访问始终是威胁程序稳定性的... 目录引言一、vector越界的底层原理与危害1.1 越界访问的本质原因1.2 越界访问的实际危害二、基

Python学习笔记之getattr和hasattr用法示例详解

《Python学习笔记之getattr和hasattr用法示例详解》在Python中,hasattr()、getattr()和setattr()是一组内置函数,用于对对象的属性进行操作和查询,这篇文章... 目录1.getattr用法详解1.1 基本作用1.2 示例1.3 原理2.hasattr用法详解2.

c++日志库log4cplus快速入门小结

《c++日志库log4cplus快速入门小结》文章浏览阅读1.1w次,点赞9次,收藏44次。本文介绍Log4cplus,一种适用于C++的线程安全日志记录API,提供灵活的日志管理和配置控制。文章涵盖... 目录简介日志等级配置文件使用关于初始化使用示例总结参考资料简介log4j 用于Java,log4c

C++归并排序代码实现示例代码

《C++归并排序代码实现示例代码》归并排序将待排序数组分成两个子数组,分别对这两个子数组进行排序,然后将排序好的子数组合并,得到排序后的数组,:本文主要介绍C++归并排序代码实现的相关资料,需要的... 目录1 算法核心思想2 代码实现3 算法时间复杂度1 算法核心思想归并排序是一种高效的排序方式,需要用

C++11范围for初始化列表auto decltype详解

《C++11范围for初始化列表autodecltype详解》C++11引入auto类型推导、decltype类型推断、统一列表初始化、范围for循环及智能指针,提升代码简洁性、类型安全与资源管理效... 目录C++11新特性1. 自动类型推导auto1.1 基本语法2. decltype3. 列表初始化3

C++11右值引用与Lambda表达式的使用

《C++11右值引用与Lambda表达式的使用》C++11引入右值引用,实现移动语义提升性能,支持资源转移与完美转发;同时引入Lambda表达式,简化匿名函数定义,通过捕获列表和参数列表灵活处理变量... 目录C++11新特性右值引用和移动语义左值 / 右值常见的左值和右值移动语义移动构造函数移动复制运算符

C++中detach的作用、使用场景及注意事项

《C++中detach的作用、使用场景及注意事项》关于C++中的detach,它主要涉及多线程编程中的线程管理,理解detach的作用、使用场景以及注意事项,对于写出高效、安全的多线程程序至关重要,下... 目录一、什么是join()?它的作用是什么?类比一下:二、join()的作用总结三、join()怎么