UOJ #11 - 【UTR #1】ydc的大树(换根 dp)
Emmm……这题似乎做法挺多的,那就提供一个想起来写起来都不太困难的做法吧。
首先不难想到一个时间复杂度 \(\mathcal O(n^2)\) 的做法:对于每个黑点我们以它为根求出离它距离最远的点集 \(S\),那么一个白点能够摧毁这个黑点当且仅当这个白点在黑点到点集 \(S\) 中的点的 \(\text{LCA}\) 的路径上。这样我们就可以求出所有白点的答案了。
考虑优化这个过程,注意“以每个点为根”一脸可用换根 \(dp\) 优化的亚子,因此考虑换根 \(dp\),如果单纯地求到每个点距离最远的黑点那你肯定会求的欸(yyq 附体),直接一遍常规的换根 \(dp\) 就完事了。不过此题还需求 \(\text{LCA}\),因此考虑以下做法,我们先以 \(1\) 为根一遍 DFS,对于每个点 \(x\) 求出以 \(x\) 为根的子树内里 \(x\) 最远的点的 \(\text{LCA}\),然后我们再额外记录两个数组 \(dis\_out_x\) 表示去掉以 \(x\) 为根的子树内剩余部分离 \(x\) 最远的黑点离 \(x\) 的距离,以及 \(lca\_out_i\) 表示它们的 \(\text{LCA}\),怎样求这两个数组呢?就按照第二遍 DFS 的套路从上往下更新,当 DFS 到 \(x\) 时将 \(x\) 的每个子树的信息压入一个 multiset,并枚举 \(x\) 的每个儿子 \(y\),将 \(y\) 的信息从 multiset 中暂时删除,如果 multiset 中距离最大值和次大值相等那么 \(lca\_out_y\) 就是 \(x\),否则 \(lca\_out_y\) 就是 multiset 中最大值对应的 \(\text{LCA}\),最后更新答案时对于每个 \(x\) 看它子树内和子树外黑点距离其的最大值,哪边大那个 \(\text{LCA}\) 就属于那边,如果相等那么 \(\text{LCA}\) 就是 \(x\)。
最后统计答案树上差分即可,时间复杂度 \(n\log n\)。
const int MAXN=1e5;
const int LOG_N=17;
const int INF=0x3f3f3f3f;
int n,m,hd[MAXN+5],to[MAXN*2+5],val[MAXN*2+5],nxt[MAXN*2+5],ec=0;bool is[MAXN+5];
void adde(int u,int v,int w){to[++ec]=v;val[ec]=w;nxt[ec]=hd[u];hd[u]=ec;}
int fa[MAXN+5][LOG_N+2],dep[MAXN+5],dis[MAXN+5],lca_in[MAXN+5];
int dis_out[MAXN+5],lca_out[MAXN+5];
int mark[MAXN+5];
void dfs1(int x=1,int f=0){
dis[x]=(is[x]^1)*(-INF);lca_in[x]=x;fa[x][0]=f;
for(int e=hd[x];e;e=nxt[e]){
int y=to[e],z=val[e];if(y==f) continue;
dep[y]=dep[x]+1;dfs1(y,x);
if(dis[y]+z>dis[x]) dis[x]=dis[y]+z,lca_in[x]=lca_in[y];
else if(dis[y]+z==dis[x]) lca_in[x]=x;
} /*printf("%d %d %d\n",x,dis[x],lca_in[x]);*/
}
int getlca(int x,int y){
if(dep[x]<dep[y]) x^=y^=x^=y;
for(int i=LOG_N;~i;i--) if(dep[x]-(1<<i)>=dep[y]) x=fa[x][i];
if(!(x^y)) return x;
for(int i=LOG_N;~i;i--) if(fa[x][i]^fa[y][i]) x=fa[x][i],y=fa[y][i];
return fa[x][0];
}
void dfs2(int x=1,int f=0){
multiset<pii> st;
for(int e=hd[x];e;e=nxt[e]){
int y=to[e],z=val[e];
if(y==f) st.insert(mp(dis_out[x],lca_out[x]));
else st.insert(mp(dis[y]+z,lca_in[y]));
} if(is[x]) st.insert(mp(0,x));
for(int e=hd[x];e;e=nxt[e]){
int y=to[e],z=val[e];if(y==f) continue;
st.erase(st.find(mp(dis[y]+z,lca_in[y])));
if(st.empty()) dis_out[y]=(is[x])?z:(-INF),lca_out[y]=(is[x])?x:0;
else{
// printf("Node %d\n",y);
pii p=*st.rbegin();st.erase(st.find(p));dis_out[y]=z+p.fi;
// printf("%d %d\n",p.fi,p.se);
if(st.empty()||(*st.rbegin()).fi<p.fi) lca_out[y]=p.se;
else lca_out[y]=x;st.insert(p);
} st.insert(mp(dis[y]+z,lca_in[y]));dfs2(y,x);
}
}
void dfs3(int x=1,int f=0){
for(int e=hd[x];e;e=nxt[e]){
int y=to[e];if(y==f) continue;
dfs3(y,x);mark[x]+=mark[y];
}
}
int main(){
scanf("%d%d",&n,&m);for(int i=1,x;i<=m;i++) scanf("%d",&x),is[x]=1;
for(int i=1,u,v,w;i<n;i++) scanf("%d%d%d",&u,&v,&w),adde(u,v,w),adde(v,u,w);dis_out[1]=-INF;
dfs1();for(int i=1;i<=LOG_N;i++) for(int j=1;j<=n;j++) fa[j][i]=fa[fa[j][i-1]][i-1];dfs2();
for(int i=1;i<=n;i++) if(is[i]){
int y=(dis[i]==dis_out[i])?i:((dis[i]<dis_out[i])?lca_out[i]:lca_in[i]);
// printf("%d %d %d %d\n",i,dis[i],dis_out[i],y);
int z=getlca(i,y);mark[i]++;mark[y]++;mark[z]--;mark[fa[z][0]]--;
} dfs3();int mx=0,cnt=0;
for(int i=1;i<=n;i++) if(!is[i]) chkmax(mx,mark[i]);
for(int i=1;i<=n;i++) cnt+=(!is[i]&&mark[i]==mx);
printf("%d %d\n",mx,cnt);
return 0;
}
UOJ #11 - 【UTR #1】ydc的大树(换根 dp)的更多相关文章
- Codeforces 891D - Sloth(换根 dp)
Codeforces 题面传送门 & 洛谷题面传送门 换根 dp 好题. 为啥没人做/yiw 首先 \(n\) 为奇数时答案显然为 \(0\),证明显然.接下来我们着重探讨 \(n\) 是偶数 ...
- [BZOJ4379][POI2015]Modernizacja autostrady[树的直径+换根dp]
题意 给定一棵 \(n\) 个节点的树,可以断掉一条边再连接任意两个点,询问新构成的树的直径的最小和最大值. \(n\leq 5\times 10^5\) . 分析 记断掉一条边之后两棵树的直径为 \ ...
- 2018.10.15 NOIP训练 水流成河(换根dp)
传送门 换根dp入门题. 貌似李煜东的书上讲过? 不记得了. 先推出以1为根时的答案. 然后考虑向儿子转移. 我们记f[p]f[p]f[p]表示原树中以ppp为根的子树的答案. g[p]g[p]g[p ...
- 换根DP+树的直径【洛谷P3761】 [TJOI2017]城市
P3761 [TJOI2017]城市 题目描述 从加里敦大学城市规划专业毕业的小明来到了一个地区城市规划局工作.这个地区一共有ri座城市,<-1条高速公路,保证了任意两运城市之间都可以通过高速公 ...
- 小奇的仓库:换根dp
一道很好的换根dp题.考场上现场yy十分愉快 给定树,求每个点的到其它所有点的距离异或上m之后的值,n=100000,m<=16 只能线性复杂度求解,m又小得奇怪.或者带一个log像kx一样打一 ...
- 国家集训队 Crash 的文明世界(第二类斯特林数+换根dp)
题意 题目链接:https://www.luogu.org/problem/P4827 给定一棵 \(n\) 个节点的树和一个常数 \(k\) ,对于树上的每一个节点 \(i\) ,求出 \( ...
- Acesrc and Travel(2019年杭电多校第八场06+HDU6662+换根dp)
题目链接 传送门 题意 两个绝顶聪明的人在树上玩博弈,规则是轮流选择下一个要到达的点,每达到一个点时,先手和后手分别获得\(a_i,b_i\)(到达这个点时两个人都会获得)的权值,已经经过的点无法再次 ...
- bzoj 3566: [SHOI2014]概率充电器 数学期望+换根dp
题意:给定一颗树,树上每个点通电概率为 $q[i]$%,每条边通电的概率为 $p[i]$%,求期望充入电的点的个数. 期望在任何时候都具有线性性,所以可以分别求每个点通电的概率(这种情况下期望=概率 ...
- codeforces1156D 0-1-Tree 换根dp
题目传送门 题意: 给定一棵n个点的边权为0或1的树,一条合法的路径(x,y)(x≠y)满足,从x走到y,一旦经过边权为1的边,就不能再经过边权为0的边,求有多少边满足条件? 思路: 首先,这道题也可 ...
随机推荐
- 解决Mybatis 报错Invalid bound statement (not found)
解决Mybatis 报错Invalid bound statement (not found) 出现此错误的原因 1.xml文件不存在 2.xml文件和mapper没有映射上 namespace指定映 ...
- 力扣 - 剑指 Offer 45. 把数组排成最小的数
题目 剑指 Offer 45. 把数组排成最小的数 思路1 将整数数组转化成字符串数组 然后使用Arrays工具类的sort方法帮助我们排序 代码 class Solution { public St ...
- 【Java虚拟机6】Java内存模型(Java篇)
什么是Java内存模型 <Java虚拟机规范>中曾试图定义一种"Java内存模型"(Java Memory Model,JMM)来屏蔽各种硬件和操作系统的内存访问差异, ...
- 【Python从入门到精通】(二)怎么运行Python呢?有哪些好的开发工具(PyCharm)
您好,我是码农飞哥,感谢您阅读本文,欢迎一键三连哦. 这是Pyhon系列文章的第二篇,本文主要介绍如何运行Python程序以及安装PyCharm开发工具. 干货满满,建议收藏,需要用到时常看看. 小伙 ...
- redis5集群搭建步骤
通常情况下为了redis的高可用,我们一般不会使用redis的单实例去运行,一般都会搭建一个 redis 的集群去运行.此处记录一下 redis5 以后 cluster 集群的搭建. 一.需求 red ...
- segyio库的使用
最近在使用segyio库读取segy文件的时候默认读取总是出现问题,经过分析发现是我们通常所用的segy格式与本库的默认格式略有不同,修改参数就可以读取: 1) with segyio.open(fi ...
- Python触发异常
我们可以使用raise语句自己触发异常,raise语法格式如下: raise [Exception [, args [, traceback]]] 语句中 Exception 是异常的类型(例如,Na ...
- 算法:N-皇后问题
一.八皇后问题 八皇后问题是一个以国际象棋为背景的问题:如何能够在8 × 8 的国际象棋棋盘上放置八个皇后(Queen),使得任何一个皇后都无法直接吃掉其他的皇后.为了达到此目的,任两个皇后都不能处于 ...
- poj 2311 Cutting Game (SG)
题意: 有一张W*H的纸片. 每人每次可以横着撕或者竖着撕,先撕出1*1那一方胜. 数据范围: W and H (2 <= W, H <= 200) 思路: 很好抽象出游戏图的模型,用SG ...
- 前端面试手写代码——JS函数柯里化
目录 1 什么是函数柯里化 2 柯里化的作用和特点 2.1 参数复用 2.2 提前返回 2.3 延迟执行 3 封装通用柯里化工具函数 4 总结和补充 1 什么是函数柯里化 在计算机科学中,柯里化(Cu ...