【做题】agc008f - Black Radius——计数&讨论&思维
原文链接 https://www.cnblogs.com/cly-none/p/9794411.html
\newcommand{\stis}[2]{\left\{ \begin{matrix} #1 \\ #2 \end{matrix} \right\}}
\newcommand{\comb}[2]{\left( \begin{matrix} #1 \\ #2 \end{matrix} \right)}
\newcommand{\floor}[1]{\left\lfloor #1 \right\rfloor}
\newcommand{\ceil}[1]{\left\lceil #1 \right\rceil}
\]
题意:给出一棵有\(n\)个结点的树\(T = \{V,E\}\)和\(V\)的一个子集\(U\)。定义一个结点的集合\(S\)合法当且仅当\(S\)能表示为\(\left\{ y \ | \ y \in V, \, dis(x,y) \leq d \right\}\)的形式,其中\(x \in U, d \in N\)。求一共有多少个合法的集合。
\(n \leq 10^5\)
先考虑\(V = U\)的情况。
首先,一个合法集合\(S\)可能被多组\((x,d)\)所表达。要对\(S\)进行计数,就必须规定某个额外限制,让它只能被一组\((x,d)\)表达。然后,通过对\((x,d)\)计数得到合法\(S\)的数量。
很容易想到\(S\)所表示的树上联通块的直径。当然,直径有可能不是唯一的。设这个长度是\(l\)。
- \(l\)为偶数。那么,能确定这个联通块的所有直径有一个共同的中点,设为\(x\)。显然\(x\)存在,且是唯一的,且\(S\)就能被\((x,\frac l 2)\)所表达。换句话说,对于每个\(x\),只要\(d\)保证\(x\)是唯一的中点,那么所有\((x,d)\)与所有\(l\)为偶数的合法集合\(S\)是一一对应的。更确切地说,就是要求\(d\)不超过以\(x\)为根时,\(x\)所有儿子中第二深的子树深度。
- \(l\)为奇数。我们发现,\(S\)的直径的中点在一条边上。记离这个中点最近的结点为\(x\)和\(y\)。这就比较麻烦了。我们不妨从另一角度考虑:\(\ceil {\frac l 2}\)一定等于以\(x\)为根时\(x\)所有儿子中第二深的子树深度+1。考虑以\(x\)为根的情况,则\(y\)为\(x\)子树深度最大的孩子结点。\((x,\ceil {\frac l 2})\)所表示的集合,与\((y, \ceil {\frac l 2}\)所表示的集合相同,等价于\(y\)的子树中没有有到\(y\)距离大于等于\(\ceil {\frac l 2}\)的结点,也就是\((x, \ceil {\frac l 2}\)覆盖了整棵树。否则\((x,\ceil {\frac l 2}\)又能唯一确定一个合法集合。
整理一下。对于每个结点\(x\),令\(D\)为以\(x\)为根时,\(x\)所有儿子中第二深的子树深度。则取\(d \leq D\)时,能表示所有直径为偶数的合法集合。假如取\(d = D + 1\)不会覆盖整棵树,那么还有一个合法集合。最后再考虑有没有少算了整棵树的情况。(假如这个树的直径为偶数,那么它已经被算过了)简单起见,前面的计算全部不计覆盖整棵树的情况,最后再+1就可以了。
for each x in V:
let d1 to be the first largest deepth of subtrees of x
let d2 to be the second largest deepth of subtrees of x
ans = ans + d2 + 1
if d1 > d2 + 1:
ans = ans + 1
if d1 == d2:
ans = ans - 1
ans = ans + 1
这样,通过\(O(n)\)树dp求出所有d1和d2,我们就完成了\(V = U\)的部分分。
然后就是\(U \subsetneq V\)的情况。称\(U\)中的结点为关键点。
如果要重新考虑每个\(x\)的\(d\)能取到多少,相当棘手。事实上,上面不少结论都会变成错误的。
又要转换一下思路。对于一个非关键点\(x\),如果\((x,d_0)\)能被某个关键点表达出来,那么,原来所有被计数的\((x,d), \, d \geq d_0\)也同样能被表达出来。记对于\(x\)最小的\(d_0\)为\(low_x\)。
然后就是要求所有\(low_x\)了。设表达出\((x,low_x)\)的关键点为\(y\)。考虑以\(x\)为根,那么\(low_x\)一定不小于\(y\)所在子树的最大深度。但只要\(low_x\)超过这个深度,\((x,low_x)\)就能被\((y,low_x + dis(x,y))\)表示出来,且按照原来的方法,这个\((y,low_x+dis(x,y))\)是不会被计数的因此,故不用担心算重的问题。那么,\(low_x\)就是以\(x\)为根时,含有关键点的深度最小的\(x\)的孩子的子树。同样可以\(O(n)\)求出。
#include <bits/stdc++.h>
using namespace std;
const int N = 200010, INF = 0x3f3f3f3f;
struct edge {
int la,b;
} con[N << 1];
int tot,fir[N];
void add(int from,int to) {
con[++tot] = (edge) {fir[from],to};
fir[from] = tot;
}
int n,mxdep[N][2],hson[N],sz[N],low[N];
char s[N];
long long ans;
void dfs(int pos,int fa) {
sz[pos] = s[pos] - '0';
mxdep[pos][0] = 0;
mxdep[pos][1] = - INF;
hson[pos] = 0;
for (int i = fir[pos] ; i ; i = con[i].la) {
if (con[i].b == fa) continue;
dfs(con[i].b,pos);
sz[pos] += sz[con[i].b];
if (mxdep[con[i].b][0] > mxdep[hson[pos]][0] || (!hson[pos]))
hson[pos] = con[i].b;
if (mxdep[con[i].b][0]+1 > mxdep[pos][0]) {
mxdep[pos][1] = mxdep[pos][0];
mxdep[pos][0] = mxdep[con[i].b][0] + 1;
} else mxdep[pos][1] = max(mxdep[pos][1],mxdep[con[i].b][0] + 1);
if (sz[con[i].b]) low[pos] = min(low[pos],mxdep[con[i].b][0] + 1);
}
}
void fsd(int pos,int fa) {
for (int i = fir[pos] ; i ; i = con[i].la) {
if (con[i].b == fa) continue;
int val = (con[i].b == hson[pos] ? mxdep[pos][1] : mxdep[pos][0]) + 1;
if (val > mxdep[con[i].b][0]) {
mxdep[con[i].b][1] = mxdep[con[i].b][0];
mxdep[con[i].b][0] = val;
hson[con[i].b] = pos;
} else mxdep[con[i].b][1] = max(mxdep[con[i].b][1],val);
if (sz[1] - sz[con[i].b] > 0)
low[con[i].b] = min(low[con[i].b],val);
fsd(con[i].b,pos);
}
}
int main() {
int x,y;
scanf("%d",&n);
for (int i = 1 ; i < n ; ++ i) {
scanf("%d%d",&x,&y);
add(x,y);
add(y,x);
}
scanf("%s",s+1);
memset(low,0x3f,sizeof low);
dfs(1,0);
fsd(1,0);
for (int i = 1 ; i <= n ; ++ i) {
int tmp = mxdep[i][1] + 1;
if (mxdep[i][0] > mxdep[i][1] + 1)
++ tmp;
if (mxdep[i][0] == mxdep[i][1]) -- tmp;
if (s[i] - '0' == 0) tmp -= low[i];
ans += max(tmp,0);
}
printf("%lld\n",ans+1);
return 0;
}
小结:这类复杂的思维题高度要求思维的清晰,思路一不留神就会混乱。清晰的思维,这确乎需要长期的锻炼。
【做题】agc008f - Black Radius——计数&讨论&思维的更多相关文章
- 【做题】CSA49F - Card Collecting Game——思维&dp
原文链接 https://www.cnblogs.com/cly-none/p/CSA49F.html 题意:Alice和Bob在玩游戏.有\(n\)种卡牌,每种卡牌有\(b_i\)张,保证\(\su ...
- 【做题】agc006e - Rotate 3x3——分析&思维
原文链接 https://www.cnblogs.com/cly-none/p/9800105.html 题意:给出一个三行\(n\)列的矩阵.问它能否由满足\(a_{ij} = 3(j-1) + i ...
- [Hdu-5155] Harry And Magic Box[思维题+容斥,计数Dp]
Online Judge:Hdu5155 Label:思维题+容斥,计数Dp 题面: 题目描述 给定一个大小为\(N*M\)的神奇盒子,里面每行每列都至少有一个钻石,问可行的排列方案数.由于答案较大, ...
- 2017国家集训队作业[agc008f]Black Radius
2017国家集训队作业[agc008f]Black Radius 时隔4个月,经历了省赛打酱油和中考各种被吊打后,我终于回想起了我博客园的密码= = 题意: 给你一棵树,树上有若干个关键点.选中某 ...
- SDOI2016 R1做题笔记
SDOI2016 R1做题笔记 经过很久很久的时间,shzr终于做完了SDOI2016一轮的题目. 其实没想到竟然是2016年的题目先做完,因为14年的六个题很早就做了四个了,但是后两个有点开不动.. ...
- AtCoder Grand Contest 1~10 做题小记
原文链接https://www.cnblogs.com/zhouzhendong/p/AtCoder-Grand-Contest-from-1-to-10.html 考虑到博客内容较多,编辑不方便的情 ...
- [Agc008F]Black Radius
[AGC008F] Black Radius Description 给你一棵有N个节点的树,节点编号为1到N,所有边的长度都为1 "全"对某些节点情有独钟,这些他喜欢的节点的信息 ...
- NOIP2016考前做题(口胡)记录
NOIP以前可能会持续更新 写在前面 NOIP好像马上就要到了,感觉在校内训练里面经常被虐有一种要滚粗的感觉(雾.不管是普及组还是提高组,我都参加了好几年了,结果一个省一都没有,今年如果还没有的话感觉 ...
- POI做题记录
嘿嘿,偷学一波! 由于博主做的题比较少,所以没按年份整理,直接按照做题时间放上来了. 2020年9月20日 [POI2013]LUK-Triumphal arch 给你一颗\(n\)个点的树(\(n\ ...
随机推荐
- python base64加密文本内容(2)
上面一篇简单进行了base64加密 但安全系数非常低,只需要用网上的在线base64解码就能破解 这里我们可以自己修改一下base64加密后的字符串,在解码的时候只需要先提前将修改的内容还原就可以了 ...
- 擠出線寬(Extrusion width),要怎麼設定?
擠出線寬(Extrusion width),要怎麼設定? Slic3r的作者,把這邊的%設定,跟"層高"做連結.我個人認為擠出線寬,要以噴頭孔徑當做設定參考才好.層高應該只要設定成 ...
- sitecore系统教程之部署架构方式分析
当您第一次部署Sitecore体验平台时,您可以选择三种主要体系结构选项: 内部部署服务器解决方案 混合服务器方案 云服务器解决方案 您是选择将Sitecore作为云,内部部署还是混合解决方案运行,取 ...
- Gitlab注册时报错:There was an error with the reCAPTCHA. Please solve the reCAPTCHA again.
今天注册时碰到以下问题: 上面的错误是因为注册时有一个google的验证码需要输入.但是中国无法访问google,因此无法访问并输入该验证码导致. 解决方案: FanQiang或者通过Github登陆 ...
- GCD (RMQ + 二分)
RMQ存的是区间GCD,然后遍历 i: 1->n, 然后不断地对[i, R]区间进行二分求以i为起点的相同gcd的区间范围,慢慢缩减区间. #include<bits/stdc++.h&g ...
- 举例说明Unicode 和UTF-8之间的转换
1)写这篇博客的原因 首先我要感谢这篇博客,卡了很久,看完下面这篇博客终于明白Unicode怎么转换成UTF-8了. https://blog.csdn.net/qq_32252957/article ...
- MQ(转)
1. 到底什么时候该使用MQ? 1). 典型场景一:数据驱动的任务依赖 采用MQ的优点是: a. 不需要预留buffer,上游任务执行完,下游任务总会在第一时间被执行 b. 依赖多个任务,被多个任务依 ...
- jar包读取jar包内部和外部的配置文件,springboot读取外部配置文件的方法
jar包读取jar包内部和外部的配置文件,springboot读取外部配置文件的方法 用系统属性System.getProperty("user.dir")获得执行命令的目录(网上 ...
- Python+OpenCV图像处理(二)——打印图片属性、设置图片存储路径、电脑摄像头的调取和显示
一. 打印图片属性.设置图片存储路径 代码如下: #打印图片的属性.保存图片位置 import cv2 as cv import numpy as np #numpy是一个开源的Python科学计算库 ...
- 为什么要使用yocto
作为灵活多变且经济高效的解决方案,嵌入式 Linux展现了巨大的价值,并广泛应用于消费电子设备.网络设备.零售点和行业应用程序.然而,广泛的应用也意味着多样化的业务需求,嵌入式解决方案开发人员必须构建 ...