set 容器详解 附大根堆题解
PART 1 set
什么是 set

——来源cppreference
简言之,它就是一种存进去就可以自动按升序排列的特殊容器,通常的 set 还具有自动去重的功能。
定义方式:
std::set<int>s;
'set'+<存储数据类型>+容器名
注意的是,这里存储数据类型不仅包含常用的int,long long,double等数值类,也可以是char,string这些字符类的,排序标准为字典序(强者处处是惊喜)。
怎么使用 set
首先,我们要了解几个 set 的常用函数:
指针类
s.begin()//指向 s 的起始
s.end()//指向 s 的末尾
容量类
s.empty()//检查容器是否为空
s.size()//返回元素数(返回值类型为 unsigned int )
修改类
s.insert(w)//向容器中插入 w 这个元素
s.erase(w)//在容器中删除 w 这个元素
s.clear()//将 s 清空
swap(s1,s2)//交换 s1 与 s2 两个容器内的元素
查找类
s.find(w)//在容器中查找 w 这个元素所在的**地址**
//若不存在该元素,则返回 s.end()
s.lower_bound(w)//在容器中寻找第一个不小于 w 的元素**地址**
//若不存在则返回 s.end()
我们可以观察到,很多 set 的返回值类型都是特殊的,因此还存在一个迭代器set<int>::iterator,它是一个指向 set<int> 中元素的指针,可以通过迭代器访问 set<int> 中的元素,并能够进行迭代器运算,如自增等操作。
我们在调用 s 中的值的时候,可以如下操作:
输出 s 中的所有值
set<int>::iterator it;
for(it=s.begin();it!=s.end();it++)
cout<<*it<<' ';// *it 是获取当前迭代器指向的元素的值
自 c++11 引入了类型 auto 后,我们可以更加简便地完成上面的操作。
for(auto it:s)
cout<<it<<' ';//这里 auto 的类型等同于我们定义 s 时的数据类型,也就是 int
set 和 multiset
multiset 在 cppreference 中定义如下:

简言之, multiset 的特性有:
- 可以保留重复的元素,也就是没有自动去重的性质。
- multiset 支持插入、删除和查找操作的平均时间复杂度均为 \(O(log\,n)\),而 set 只支持插入和查找操作的平均时间复杂度为 \(O(log\,n)\),删除操作的平均时间复杂度为 \(O(1)\)。
- multiset 在删除时只会删除元素值相同的元素中的一个,而不是全部删除或删除一些。
通常情况下,我们多使用 set ,因为它在进行查找等操作时更快;而只有在需要保留重复元素的少数情况下,我们使用 multiset ,下题就是一个例子。
PART 2 大根堆
题面

思路
常规办法为线段树合并,但太长了不想写 但总感觉有更优的做法,所以就有了下面基于 set 优化的 dfs 做法。
我们可以通过 dp 引入,固定一点作为根结点,用f[u][i]表示以 \(u\) 为树根的子树里结点权值小于 \(i\) 的个数,其中 \(i\le v_u\);用 \(size_u\) 表示以 \(u\) 为根的树的大小。
那么很容易能想到状态转移方程为:
\]
\]
那么如何存储? multiset !
set 自带的查找功能可以很方便的判断条件是否成立, \(size\) 函数也直接提供了上述 \(size\) 数组。
使用 dfs 进行遍历,由子结点逐个回溯至根节点,最后根节点的 \(size\) ,即为答案。
代码中加入了部分注释,供参考。
code:
#include<bits/stdc++.h>
inline int qr()
{
char ch=getchar();int x=0,f=1;
for(;ch<'0'||ch>'9';ch=getchar())if(ch=='-')f=-1;
for(;ch>='0'&&ch<='9';ch=getchar())x=(x<<3)+(x<<1)+(ch^48);
return x*f;
}
#define qr qr()
using namespace std;
const int Ratio=0;
const int N=500005;
int n,cnt;
int va[N],hh[N],to[N],ne[N];
multiset<int>f[N];
namespace Wisadel
{
void Wadd(int u,int v)
{//加边
to[++cnt]=v;
ne[cnt]=hh[u];
hh[u]=cnt;
}
void Wdfs(int u,int fa)
{
for(int i=hh[u];i!=-1;i=ne[i])
{//遍历
int t=to[i];
if(t==fa)
continue;
Wdfs(t,u);
if(f[u].size()<f[t].size())
swap(f[u],f[t]);
//大根堆,所以父节点的子树和应大于子节点
for(auto j:f[t])
f[u].insert(j);
//类似线段树合并 将两棵树并到一起
f[t].clear();
//擦去被合并了的树
}
if(f[u].size()>0&&f[u].lower_bound(va[u])!=f[u].end())
f[u].erase(f[u].lower_bound(va[u]));
//因为是大根堆,所以子应小于父
//那么若存在比父大的元素,这个堆便不成立
//擦去它
f[u].insert(va[u]);
//把当前结点(根节点)插入
}
short main()
{
memset(hh,-1,sizeof hh);
n=qr;
for(int i=1;i<=n;i++)
{
va[i]=qr;int b=qr;
if(i!=1)
Wadd(i,b),Wadd(b,i);
}
Wdfs(1,-1);
printf("%d\n",(int)f[1].size());
//我们所用dfs的遍历形式,保证了会先将尽头的子树遍历尽
//所以每个结点遍历后的结果,是包含了它以及它子树的最优解
//所以经过交换,最后答案会体现在f[1]容器内元素的个数
return Ratio;
}
}
int main(){return Wisadel::main();}
完结撒花~

set 容器详解 附大根堆题解的更多相关文章
- 最小割树(Gomory-Hu Tree)求无向图最小割详解 附 BZOJ2229,BZOJ4519题解
最小割树(Gomory-Hu Tree) 前置知识 Gomory-Hu Tree是用来解决无向图最小割的问题的,所以我们需要了解无向图最小割的定义 和有向图类似,无向图上两点(x,y)的割定义为一个边 ...
- linux管道命令grep命令参数及用法详解---附使用案例|grep
功能说明:查找文件里符合条件的字符串. 语 法:grep [-abcEFGhHilLnqrsvVwxy][-A<显示列数>][-B<显示列数>][-C<显示列数>] ...
- [Spring学习笔记 1 ] Spring 简介,初步知识--Ioc容器详解 基本原理。
一.Spring Ioc容器详解(1) 20131105 1.一切都是Bean Bean可是一个字符串或者是数字,一般是一些业务组件. 粒度一般比较粗. 2.Bean的名称 xml配置文件中,id属性 ...
- C++ STL bitset 容器详解
C++ STL bitset 容器详解 本篇随笔讲解\(C++STL\)中\(bitset\)容器的用法及常见使用技巧. \(bitset\)容器概论 \(bitset\)容器其实就是个\(01\)串 ...
- 跟我一起学STL(2)——vector容器详解
一.引言 在上一个专题中,我们介绍了STL中的六大组件,其中容器组件是大多数人经常使用的,因为STL容器是把运用最广的数据结构实现出来,所以我们写应用程序时运用的比较多.然而容器又可以序列式容器和关联 ...
- Java 序列化Serializable详解(附详细例子)
Java 序列化Serializable详解(附详细例子) 1.什么是序列化和反序列化Serialization(序列化)是一种将对象以一连串的字节描述的过程:反序列化deserialization是 ...
- 2.3 C++STL vector容器详解
文章目录 2.3.1 引入 2.3.2 代码实例 2.3.3 运行结果 总结 2.3.1 引入 vector 容器 动态数组 可变数组 vector容器 单口容器(尾部操作效率高) vector动态增 ...
- Windows WMIC命令使用详解(附实例)
第一次执行WMIC命令时,Windows首先要安装WMIC,然后显示出WMIC的命令行提示符.在WMIC命令行提示符上,命令以交互的方式执行 执行“wmic”命令启动WMIC命令行环境.这个命令可以在 ...
- SILC超像素分割算法详解(附Python代码)
SILC算法详解 一.原理介绍 SLIC算法是simple linear iterative cluster的简称,该算法用来生成超像素(superpixel) 算法步骤: 已知一副图像大小M*N,可 ...
- [No000013A]Windows WMIC命令使用详解(附实例)
第一次执行WMIC命令时,Windows首先要安装WMIC,然后显示出WMIC的命令行提示符.在WMIC命令行提示符上,命令以交互的方式执行 执行“wmic”命令启动WMIC命令行环境.这个命令可以在 ...
随机推荐
- CF-937(D,E)
CF-937 在补题-- D 分析 我们发现这些因子都是二进制形式的十进制数,n的范围是$1e5$,16的二进制是$10000$,于是可以枚举1~16,把这些因子预处理出来,对于每个n就枚举因子再作除 ...
- #模拟#U137456 数字
题目 牛牛和他的小伙伴们高高兴兴的吃完了蛋糕,吃完蛋糕之后就到了牛牛和他的小伙伴们最喜欢的环节了--猜数 字, 这次是牛牛的生日,大家决定让牛牛来制定规则,由于牛牛的生日是4月7日,所以牛牛特别喜欢数 ...
- OpenHarmony标准系统开机时长优化
简介 万物互联时代,产品性能至关重要,而系统启动时间是系统性能的重要组成部分,因为用户必须等待系统启动完成后才能使用设备.对于经常需要进行冷启动的汽车等设备而言,较短的启动时间至关重要(没有人喜欢在等 ...
- 部署解压版mysql
1.检查系统是否安装过mysql //检查系统中有无安装过mysql rpm -qa|grep mysql //查询所有mysql 对应的文件夹,全部删除 whereis mysql find / - ...
- char * 、BSTR、long、wchar_t *、LPCWSTR、string、QString、CStringA类型转换
char* 转 BSTR char* s1 = "zhangsan"; CString s2 = CString(s1); BSTR s3 = s2.AllocSysString( ...
- 前端使用 Konva 实现可视化设计器(3)
github/gitee Star 终于有几个了! 从这章开始,难度算是(或者说细节较多)升级,是不是值得更多的 Star 呢?! 继续求 Star ,希望大家多多一键三连,十分感谢大家的支持~ 创作 ...
- 推荐几款Vue后台管理系统的框架,以便备用
推荐几款Vue后台管理系统的框架,以便备用 Vue.js 是一个目前比较流行的前端框架,在业界也算很有名气,今天这里为大家罗列一下基于Vue的后端管理的框架. 使用这些框架你会发现它包括了我们常用的路 ...
- 多python版本的库安装和导库
同时安装多python版本的,使用pip安装python的库,以及导出python库列表及版本,使用导出的库列表批量进行新环境的库安装. 1.同时安装python2和python3时,要进行pip安装 ...
- json文件读取并转换成为字典python
# JSON到字典转化 f2 = open('info.json', 'r') info_data = json.load(f2) print(info_data) # 显示数据类型 print(ty ...
- 力扣319(java)-灯泡开关(中等)
题目: 初始时有 n 个灯泡处于关闭状态.第一轮,你将会打开所有灯泡.接下来的第二轮,你将会每两个灯泡关闭第二个. 第三轮,你每三个灯泡就切换第三个灯泡的开关(即,打开变关闭,关闭变打开).第 i 轮 ...
