<STL - 2> 集合set 与 multiset

一. 介绍

1. set的介绍

◉ set是一个集合容器,其中所包含的元素是唯一的,集合中的元素按一定的顺序排列。元素插入过程是按排序规则插入,所以不能指定插入位置

◉ set采用红黑树变体的数据结构实现,红黑树属于平衡二叉树。在插入操作和删除操作上比vector快

◉ set不可以直接存取元素(不可以使用at.(pos)与[]操作符

刚好复习一题需要排序和去重的题
P1059 [NOIP2006 普及组] 明明的随机数

2 . multset的介绍

◉ multiset与set的区别:set支持唯一键值,每个元素值只能出现一次(去重的原理);而multiset中同一值可以出现多次

不可以直接修改set或multiset容器中的元素值,因为该类容器是自动排序的。如果希望修改一个元素值,必须先删除原有的元素,再插入新的元素

特点

实际运用很多时候是利用去重功能的一个类
set multiset 都在头文件 < set >

写这个过程中刚好写到一题
P2676 [USACO07DEC] Bookshelf B

二. 常用函数

1. 默认构造

set 集合名字;
mutiset 集合名字;

set.insert(elem); //在容器中插入元素
set.begin(); //返回容器中第一个数据的迭代器
set.end();//返回容器中最后一个数据之后的迭代器
set.rbegin(); //返回容器中倒数第一个元素的迭代器
set.rend(); //返回容器中倒数最后一个元素的后面的迭代器

2. set容器的构造并初始化

set(另一个set); //把一个集合拷贝到另一个集合中的拷贝构造函数

set.swap(另一个集合); //交换两个集合容器

3. 与set的大小有关的函数

set.size(); //返回容器中元素的数目

set.empty();//判断容器是否为空

显然不能饿resize()

4. set中元素的删除

set.clear(); //清除所有元素

set.erase(pos); //删除pos迭代器所指的元素,返回下一个元素的迭代器

set.erase(beg,end); //删除区间 [beg,end) 的所有元素

set.erase(elem); //删除容器中值所有为elem的元素,当然只会有一个

5. set中元素的排序 (重)

vs2022中set容器默认是升序排序

基础语法

set<typename, less > 集合名;//less表示按照升序方式排序元素

注 : 类型set 相当于类型set<typename, less >

set<typename, greater > 集合名;//greater表示按照降序方式排序元素

其中less 和 greater中的类型可以任意, 该类型要和set容纳的数据类型一致

注意: 千万记得 less<>和greater<>中的类型

6. set中元素的查找

①. set.find(elem);

查找elem元素,返回指向elem元素的迭代器

    set<int>::iterator it1; // 迭代器的类型是iterator

    it1 = s1.find(10);// 如果查找的数不在set中,会返回end()set末尾的迭代器;

    it1--;

    cout << *it1 << endl; // 确实得到最后一个元素

find()返回的是iterator的对象, 名字是find函数的名字 , 类型<T&x> 返回的是iterator find <T&x>而不是真的像指针一样返回Null

②. set.count(elem);

返回容器中值为elem的元素个数。对set来说,要么是0,要么是1。对multiset来说,值可能大于1

其中0表示不在, 1表示在
例题 : P2141 [NOIP2014 普及组] 珠心算测验

③.set.lower_bound(elem);

返回第一个 >=elem 元素的 迭代器, 相当于得到一个分界线且包括分界线本身

④.set.upper_bound(elem);

返回第一个>elem元素的迭代器, 不包括分界线本身

//有一个集合1 3 5 7 8 10
auto it2 = s1.lower_bound(5);

for (; it2 != s1.end(); it2++)

        cout << *it2<< ' '; // 5 8 7 10

cout << endl;



auto it3 = s1.upper_bound(5);

for (; it3 != s1.end(); it3++)

        cout << *it3 << ' '; //8 7 10

cout << endl;

对比 :
lower_bound(elem) VS set.upper_bound(elem)

相比之下, upper就比lower少了一个不包含自身
可以认为upper是开区间, lower 是闭区间

⑤. 奇怪的查找函数set.equal_range(elem)

●语法 :
set.equal_range(elem), 返回容器中与elem相等的上下限的两个迭代器。上限是闭区间,下限是开区间,如[beg,end)

函数返回两个迭代器,而这两个迭代器被封装在 pair(对) 中, 一个pair中可以存储两种数据

例如 : <int,char> <int, string> <set::iterator , set::iterator>

这两个类型也可以是自定义类型
●例 1 :

pair<set<int>::iterator, set<int>::iterator> p = s1.equal_range(7);
// 接收返回的迭代器

 cout << (* p.first) << endl; 
//由结果可知 first 是对应两个迭代器中的 左闭迭代器 beg 可以理解为 lower_bound 的返回值

cout << (* p.second) << endl;
//而 second 是对应两个迭代器中的 右开迭代器end 同样可以理解为upper_bound 的返回值

●例 2 :

pair<set<int>::iterator, set<int>::iterator> p1 = s1.equal_range(11);



cout << endl;

        set<int>::iterator it4 = p1.first;

        set<int>::iterator it5 = p1.second;

        it4--;

        it5--;

cout << *it4 << endl; 

cout << *it5 << endl;   //由两值可知, 两个迭代器 '自减前' 指向的都是end()

//如果传入11, 是不存在的元素 , 那么根据编译器不同 , 返回的值会不同, vs2022中两个迭代器都是返回 set 的end() , 通过 '--' 可以得到最后一个元素
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值