[CF768G] The Winds of Winter
Discription:
断开树的每一个点会形成一个森林,然后可以进行一次操作:将森林中的一棵树接到另一棵树上。使得森林中size最大的树size最小。依次输出对于每个结点的最小size值
Hint:
\(n<=1e5\)
Solution:
用multiset维护每个点的size域,每次贪心将最大的树的子树接到最小的树上,答案用二分答案求出
本题细节较多,需要处理各种情况
#include<bits/stdc++.h>
using namespace std;
typedef multiset<int >::iterator mit;
const int mxn=1e5+5;
struct ed {
int to,nxt;
}t[mxn];
int n,rt,cnt;
int hd[mxn],sz[mxn],f[mxn],son[mxn],ans[mxn];
multiset<int >mp[mxn],oth,anc;
inline void add(int u,int v) {
t[++cnt]=(ed) {v,hd[u]}; hd[u]=cnt;
}
void dfs1(int u)
{
sz[u]=1;
for(int i=hd[u];i;i=t[i].nxt) {
int v=t[i].to;
dfs1(v); sz[u]+=sz[v];
if(sz[v]>sz[son[u]]) son[u]=v;
}
if(u!=rt) oth.insert(sz[u]);
} //第一遍Dfs预处理出重儿子和初始size
inline int check(int opt,int val,int u,int mi,int mx)
{
if(opt) {
mit tp=mp[son[u]].lower_bound(mx-val);
if(tp!=mp[son[u]].end()&&(*tp)+mi<=val) return 1;
}
else {
mit tp=oth.lower_bound(mx-val);
if(tp!=oth.end()&&(*tp)+mi<=val) return 1;
tp=anc.lower_bound(mx-val+sz[u]);
if(tp!=anc.end()&&(*tp)+mi-sz[u]<=val) return 1;
}
return 0;
} //分类讨论检验答案
void dfs2(int u)
{
if(u!=rt) oth.erase(oth.lower_bound(sz[u])); //一开始删除,后面会加进来
if(f[u]&&f[u]!=rt) anc.insert(sz[f[u]]);
//因为祖先size是变化的,故将祖先单独拿出来维护
int mx1=max(n-sz[u],sz[son[u]]); //子树size的最大值
int mx2=min(n-sz[u],sz[son[u]]); //子树size的次大值
int mi=n-sz[u]; //子树size的最小值
if(!mi) mi=sz[son[u]];
for(int i=hd[u];i;i=t[i].nxt) {
int v=t[i].to;
if(v==son[u]) continue ;
dfs2(v);
for(mit it=mp[v].begin();it!=mp[v].end();++it)
oth.insert(*it);
mi=min(mi,sz[v]),mx2=max(mx2,sz[v]);
}
if(son[u]) dfs2(son[u]),mi=min(mi,sz[son[u]]);
for(int i=hd[u];i;i=t[i].nxt) {
int v=t[i].to;
if(v==son[u]) continue ;
for(mit it=mp[v].begin();it!=mp[v].end();++it)
oth.erase(oth.lower_bound(*it));
}
if(mx1!=mx2) {
int l=mx2,r=mx1;
int opt=(mx1==sz[son[u]]);
while(l<=r) {
int mid=(l+r)>>1;
if(check(opt,mid,u,mi,mx1)) ans[u]=mid,r=mid-1;
else l=mid+1;
}
} //当操作有意义时,二分答案
if(!ans[u]) ans[u]=mx1;
if(son[u]) swap(mp[u],mp[son[u]]);
//每次直接O(1)继承重儿子的set,从而保证了时间复杂度
for(int i=hd[u];i;i=t[i].nxt) {
int v=t[i].to;
if(v==son[u]) continue ;
for(mit it=mp[v].begin();it!=mp[v].end();mp[v].erase(it++))
mp[u].insert(*it);
}
if(f[u]&&f[u]!=rt) anc.erase(anc.lower_bound(sz[f[u]]));
mp[u].insert(sz[u]);
}
void solve()
{
dfs2(rt);
for(int i=1;i<=n;++i) printf("%d\n",ans[i]);
}
int main()
{
int u,v;
scanf("%d",&n);
for(int i=1;i<=n;++i) {
scanf("%d %d",&u,&v);
if(!u) rt=v;
else add(u,v),f[v]=u;
}
dfs1(rt); solve();
return 0;
}
[CF768G] The Winds of Winter的更多相关文章
- 【CF768G】The Winds of Winter 可持久化线段树 DFS序
题目大意 给定一棵\(n\)个点的树,对于树上每个结点,将它删去,然后可以将得到的森林中任意一个点与其父亲断开并连接到另一颗树上,对每一个点求出森林中所有树\(size\)最大值的最小值. \(n\l ...
- Winter Storm Warning
2019-01-02 11:22:33 ...WINTER STORM WARNING REMAINS IN EFFECT UNTIL 6 AM AKST TUESDAY... * WHAT...He ...
- 开发框架Data Abstract和Hydra发布版本Winter 2013
Data Abstract Winter 2013即Data Abstract Version 7.0.73 (Build .1111),Winter 2013版对Data Abstract继续做了以 ...
- 2015 UESTC Winter Training #10【Northeastern Europe 2009】
2015 UESTC Winter Training #10 Northeastern Europe 2009 最近集训都不在状态啊,嘛,上午一直在练车,比赛时也是刚吃过午饭,状态不好也难免,下次比赛 ...
- 2015 UESTC Winter Training #8【The 2011 Rocky Mountain Regional Contest】
2015 UESTC Winter Training #8 The 2011 Rocky Mountain Regional Contest Regionals 2011 >> North ...
- 2015 UESTC Winter Training #7【2010-2011 Petrozavodsk Winter Training Camp, Saratov State U Contest】
2015 UESTC Winter Training #7 2010-2011 Petrozavodsk Winter Training Camp, Saratov State U Contest 据 ...
- 2015 UESTC Winter Training #6【Regionals 2010 >> North America - Rocky Mountain】
2015 UESTC Winter Training #6 Regionals 2010 >> North America - Rocky Mountain A - Parenthesis ...
- 2015 UESTC Winter Training #4【Regionals 2008 :: Asia - Tehran】
2015 UESTC Winter Training #4 Regionals 2008 :: Asia - Tehran 比赛开始时电脑死活也连不上WIFI,导致花了近1个小时才解决_(:зゝ∠)_ ...
- winds引导配置数据文件包含的os项目无效
我装了winds7与linux双系统,用easyBcd程序时,删除了一个winds7,之后winds7就进不去了, 进入winds7时显示winds未能启动,原因可能是最近更改了硬件或软件.解决此问题 ...
随机推荐
- ubuntu下配置反向代理
1. 环境 ubuntu:Ubuntu 13.04 x86-64 apache2: 2.2.22-6ubuntu5.1 amd64 2. 配置 2.1 配置应用 增加监听端口 打开/etc/apac ...
- 饮冰三年-人工智能-linux-02 初始Linux
参考博客:https://www.cnblogs.com/linhaifeng/articles/6045600.html 1:初始Linux命令 右击,开启终端,或者ctrl+alt[F1-F6]的 ...
- Django认证系统auth认证
使用Django认证系统auth认证 auth认证系统可以处理范围非常广泛的任务,且具有一套细致的密码和权限实现.对于需要与默认配置不同需求的项目,Django支持扩展和自定义认证;会将用户信息写入到 ...
- RazorEngine.Templating MVC中View当模板
最近在做一个生成JSON的功能,比较笨的办法就是把需要的数据拆分开,保存到数据库,在从数据库中取出来进行拼接.这种方法比较笨,代码就不贴了. 需要注意拼接的时的转义字符: "\"s ...
- OpenSSL + Windows 下载安装
Download Win32 OpenSSLhttps://slproweb.com/products/Win32OpenSSL.htmlhttps://wiki.openssl.org/index. ...
- JavaScript中this 详解
涵义 this 关键字是一个非常重要的语法点.毫不夸张地说,不理解它的含义,大部分开发任务都无法完成. 首先, this 总是返回一个对象,简单说,就是返回属性或方法“当前”所在的对象. this.p ...
- Spring data JPA中使用Specifications动态构建查询
有时我们在查询某个实体的时候,给定的条件是不固定的,这是我们就需要动态 构建相应的查询语句,在JPA2.0中我们可以通过Criteria接口查询,JPA criteria查询.相比JPQL,其优势是类 ...
- [转] 设置div的overflow:scroll,但是在手机上滑动的时候有点卡顿
设置div的overflow:scroll,但是在手机上滑动的时候有点卡顿,所以在这个div上加一个css: -webkit-overflow-scrolling : touch; 在苹果手机上使用- ...
- mysql中cast() 和convert()的用法讲解
一.在mysql操作中我们经常需要对数据进行类型转换.此时我们应该使用的是cast()或convert(). 二.两者的对比 相同点:都是进行数据类型转换,实现的功能基本等同 不同点:两者的语法不同, ...
- UICollectionView的常用方法
class UICollectionView : UIScrollView //初始化,位置,风格 init(frame: CGRect, collectionViewLayout layout: U ...