声明

本文中题解部分内容大部分转载自 @sonnety这篇博客中,本文为为方便复习而写的结论类文章,读者可自行跳转至原文处阅读。


PART 1 set

什么是 set



——来源cppreference

简言之,它就是一种存进去就可以自动按升序排列的特殊容器,通常的 set 还具有自动去重的功能。

定义方式:

std::set<int>s;
'set'+<存储数据类型>+容器名

注意的是,这里存储数据类型不仅包含常用的intlong longdouble等数值类,也可以是charstring这些字符类的,排序标准为字典序(强者处处是惊喜)。

怎么使用 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 的特性有:

  1. 可以保留重复的元素,也就是没有自动去重的性质。
  2. multiset 支持插入、删除和查找操作的平均时间复杂度均为 \(O(log\,n)\),而 set 只支持插入和查找操作的平均时间复杂度为 \(O(log\,n)\),删除操作的平均时间复杂度为 \(O(1)\)。
  3. multiset 在删除时只会删除元素值相同的元素中的一个,而不是全部删除或删除一些。

通常情况下,我们多使用 set ,因为它在进行查找等操作时更快;而只有在需要保留重复元素的少数情况下,我们使用 multiset ,下题就是一个例子。

PART 2 大根堆

题面

思路

常规办法为线段树合并,但太长了不想写 但总感觉有更优的做法,所以就有了下面基于 set 优化的 dfs 做法。

我们可以通过 dp 引入,固定一点作为根结点,用f[u][i]表示以 \(u\) 为树根的子树里结点权值小于 \(i\) 的个数,其中 \(i\le v_u\);用 \(size_u\) 表示以 \(u\) 为根的树的大小。

那么很容易能想到状态转移方程为:

\[f[u][i]=\sum_{size_u}f[u][i],i\le v_u
\]
\[f[u][i]=max(\sum_{size_u}f[u][i],\sum_{size_u}f[u][i+1]-1),i\ge v_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 容器详解 附大根堆题解的更多相关文章

  1. 最小割树(Gomory-Hu Tree)求无向图最小割详解 附 BZOJ2229,BZOJ4519题解

    最小割树(Gomory-Hu Tree) 前置知识 Gomory-Hu Tree是用来解决无向图最小割的问题的,所以我们需要了解无向图最小割的定义 和有向图类似,无向图上两点(x,y)的割定义为一个边 ...

  2. linux管道命令grep命令参数及用法详解---附使用案例|grep

    功能说明:查找文件里符合条件的字符串. 语 法:grep [-abcEFGhHilLnqrsvVwxy][-A<显示列数>][-B<显示列数>][-C<显示列数>] ...

  3. [Spring学习笔记 1 ] Spring 简介,初步知识--Ioc容器详解 基本原理。

    一.Spring Ioc容器详解(1) 20131105 1.一切都是Bean Bean可是一个字符串或者是数字,一般是一些业务组件. 粒度一般比较粗. 2.Bean的名称 xml配置文件中,id属性 ...

  4. C++ STL bitset 容器详解

    C++ STL bitset 容器详解 本篇随笔讲解\(C++STL\)中\(bitset\)容器的用法及常见使用技巧. \(bitset\)容器概论 \(bitset\)容器其实就是个\(01\)串 ...

  5. 跟我一起学STL(2)——vector容器详解

    一.引言 在上一个专题中,我们介绍了STL中的六大组件,其中容器组件是大多数人经常使用的,因为STL容器是把运用最广的数据结构实现出来,所以我们写应用程序时运用的比较多.然而容器又可以序列式容器和关联 ...

  6. Java 序列化Serializable详解(附详细例子)

    Java 序列化Serializable详解(附详细例子) 1.什么是序列化和反序列化Serialization(序列化)是一种将对象以一连串的字节描述的过程:反序列化deserialization是 ...

  7. 2.3 C++STL vector容器详解

    文章目录 2.3.1 引入 2.3.2 代码实例 2.3.3 运行结果 总结 2.3.1 引入 vector 容器 动态数组 可变数组 vector容器 单口容器(尾部操作效率高) vector动态增 ...

  8. Windows WMIC命令使用详解(附实例)

    第一次执行WMIC命令时,Windows首先要安装WMIC,然后显示出WMIC的命令行提示符.在WMIC命令行提示符上,命令以交互的方式执行 执行“wmic”命令启动WMIC命令行环境.这个命令可以在 ...

  9. SILC超像素分割算法详解(附Python代码)

    SILC算法详解 一.原理介绍 SLIC算法是simple linear iterative cluster的简称,该算法用来生成超像素(superpixel) 算法步骤: 已知一副图像大小M*N,可 ...

  10. [No000013A]Windows WMIC命令使用详解(附实例)

    第一次执行WMIC命令时,Windows首先要安装WMIC,然后显示出WMIC的命令行提示符.在WMIC命令行提示符上,命令以交互的方式执行 执行“wmic”命令启动WMIC命令行环境.这个命令可以在 ...

随机推荐

  1. 一文弄懂java中的Queue家族

    目录 简介 Queue接口 Queue的分类 BlockingQueue Deque TransferQueue 总结 java中Queue家族简介 简介 java中Collection集合有三大家族 ...

  2. 将Map中对应的key和value赋值到对象中

    BeanUtils位于import org.apache.commons.beanutils.BeanUtils包下 其使用方法: Map<String, Object> objectMa ...

  3. Docker 12 Dockerfile

    简介 Dockerfile 是用来构建 Docker 镜像的文件,可以理解为命令参数脚本. Dockerfile 是面向开发的,想要打包项目,就要编写 Dockerfile 文件. 由于 Docker ...

  4. Mysql之innodb架构

    Innodb存储引擎的架构 内存结构 Bufer Pool 缓冲池是主内存中的一个区域,里面可以缓存磁盘上经常操作的真实数据,在执行增删改查操作时,先操作缓冲池中的数据(若缓冲池没有数据,则从磁盘加载 ...

  5. Prometheus之node_exporter安装

    一.简介 node_exporter用来安装到被监控的主机上,暴露被监控主机的指标数据,服务器端基于http协议调用的端口9100(默认)来获取被监控服务器信息. 二.安装部署 下载地址 https: ...

  6. Ez_pycode_dis qsnctfwp

    Python字节码基础 下载相关文件并打开,其中为 Python 字节码. 字节码格式为 源码行号 | 指令在函数中的偏移 | 指令符号 | 指令参数 | 实际参数值 根据上述字节码格式以及文件内容开 ...

  7. Centos7、CentOS8、CentOS9 修改硬盘分区大小扩充root分区大小

    Centos7 修改硬盘分区大小,实现CentOS无损分区扩容 扩充root分区大小 安装了CentOS7 开发环境及软件后,发现root分区已经才剩下不到1G空间,难不成要干掉重装,OMG,NO! ...

  8. 如何增强Java Excel API 的导入和导出性能

    前言 GrapeCity Documents for Excel (以下简称GcExcel)是葡萄城公司的一款服务端表格组件,它提供了一组全面的 API 以编程方式生成 Excel (XLSX) 电子 ...

  9. 第四課-Channel Study File Reader & File Writer

    示例描述:从数据库中读取数据并过滤转换为HL7并存放到指定目录;然后读取目录中的HL7文件转换为txt文本并存放到指定目录. 首先在F:\MirthConnect\Test目录下创建Out目录存放输出 ...

  10. 力扣619(MySQL)-只出现一次的最大数字(简单)

    题目: MyNumbers 表: 单一数字 是在 MyNumbers 表中只出现一次的数字. 请你编写一个 SQL 查询来报告最大的 单一数字 .如果不存在 单一数字 ,查询需报告 null . 查询 ...