[ARC097F]Monochrome Cat
题意:一棵树,每个节点是黑色或白色,你可以从任意节点开始进行一些操作并在任意节点结束,如果当前在$x$,那么一次操作可以是:1.走到相邻节点$y$并翻转$y$的颜色,2.翻转$x$的颜色,问把所有节点都变黑最少要多少次操作
首先当然要把黑色叶子全部删掉,这个类似拓扑排序一样做即可
然后证一个小结论:不会有一条边被经过$\gt2$次
设$(x,y)$被经过$3$次,那么操作序列为$A\Rightarrow(x\rightarrow y)\Rightarrow B\Rightarrow(y\rightarrow x)\Rightarrow C\Rightarrow(x\rightarrow y)\Rightarrow D$,我们可以把操作序列变为$A\Rightarrow C\Rightarrow\text~x\Rightarrow(x\rightarrow y)\Rightarrow\text~y\Rightarrow B\Rightarrow D$,这样等价并且没有增加操作次数
所以最优解形如:选定起点$s$和终点$t$,$s\rightarrow t$路径上的边只经过一次,其他边经过两次,并在过程中适时使用操作$2$
更进一步:存在最优解使得$s,t$都是叶子
假设起点为$x$,终点为$y$且$y$不是叶子,$z$是$y$往远离$x$方向的第一个节点,那么把原来的$(y\rightarrow z)\Rightarrow X\Rightarrow(z\rightarrow y)$变成$\text~y\Rightarrow(y\rightarrow z)\Rightarrow X$即可,同样不增加操作次数
现在考虑怎么求答案,如果$s=t$,设$d_x$为$x$的度数并记$v_x=\left[(d_x\equiv1(\bmod2),c_x=B)\text{ or }(d_x\equiv0(\bmod2),c_x=W)\right]$,那么答案是$2|E|+\sum\limits_xv_x$
当$t$移动时,$s\rightarrow t$上的边经过次数$-1$,$s\rightarrow t$这条链上$\neq t$的点(以下记这条链为$[s,t)$)的贡献也要重新计算,要加上$-\text{dis}(s,t)+\sum\limits_{x\in[s,t)}-[v_x=1]+[v_x=0]$
所以如果我们给每个点$x$一个权值$-[v_x=1]+[v_x=0]-1$,找到最小的叶子到叶子的链即可,dfs一遍统计答案即可
monochrome:单色的,黑白的
#include<stdio.h>
#include<algorithm>
using namespace std;
int h[100010],nex[200010],to[200010],M,n;
void add(int a,int b){
M++;
to[M]=b;
nex[M]=h[a];
h[a]=M;
}
int d[100010],q[100010];
bool del[100010];
char s[100010];
void topsort(){
int head,tail,x,i;
head=1;
tail=0;
for(i=1;i<=n;i++){
if(s[i]=='B'&&d[i]==1)q[++tail]=i;
}
while(head<=tail){
x=q[head++];
del[x]=1;
for(i=h[x];i;i=nex[i]){
if(d[to[i]])d[to[i]]--;
if(s[to[i]]=='B'&&d[to[i]]==1&&!del[to[i]])q[++tail]=to[i];
}
}
}
int f[100010],sum,mn;
int tp(int x){return(d[x]&1)^(s[x]=='W');}
int val(int x){return tp(x)==1?-2:0;}
void dfs(int fa,int x){
sum+=(tp(x)==1)+2;
f[x]=val(x);
for(int i=h[x];i;i=nex[i]){
if(to[i]!=fa&&!del[to[i]]){
dfs(x,to[i]);
mn=min(mn,f[x]+f[to[i]]);
f[x]=min(f[x],f[to[i]]+val(x));
}
}
}
int main(){
int i,x,y;
scanf("%d",&n);
for(i=1;i<n;i++){
scanf("%d%d",&x,&y);
add(x,y);
add(y,x);
d[x]++;
d[y]++;
}
scanf("%s",s+1);
topsort();
for(x=1;x<=n;x++){
if(!del[x])break;
}
if(x>n)
putchar('0');
else{
dfs(0,x);
sum-=2;
printf("%d",sum+mn);
}
}
[ARC097F]Monochrome Cat的更多相关文章
- Atcoder 乱做
最近感觉自己思维僵化,啥都不会做了-- ARC103 F Distance Sums 题意 给定第 \(i\) 个点到所有点的距离和 \(D_i\) ,要求构造一棵合法的树.满足第 \(i\) 个点到 ...
- 【AtCoder】ARC097 (C - F)题解
C - K-th Substring 题解 找出第K大的子串,重复的不计入 这个数据范围可能有什么暴力可以艹过去吧,但是K放大的话这就是后缀自动机板子题啊= = 代码 #include <ios ...
- AtCoder Regular Contest 097
AtCoder Regular Contest 097 C - K-th Substring 题意: 求一个长度小于等于5000的字符串的第K小子串,相同子串算一个. K<=5. 分析: 一眼看 ...
- 基于Cat的分布式调用追踪
Cat是美团点评出的一款APM工具,同类的产品也有不少,知名的开源产品如zipkin和pinpoint:国内收费的产品如oneapm.考虑到Cat在互联网公司的应用比较广,因此被纳入选型队列,我也有幸 ...
- mkdir,rmdir,cp,rm,mv,cat,touch用法
一.mkdir新建目录 1.进入tmp目录,查看该目录下面的子目录 [root@localhost ~]# cd /tmp[root@localhost tmp]# lshsperfdata_root ...
- 大众点评cat系统的搭建笔记
项目地址:https://github.com/dianping/cat 编译步骤: 这个项目比较另类,把编译需要的jar包,单独放在git分支mvn-repo里了,而且官方文档里给了一个错误的命令提 ...
- cat命令使用
cat:concatenate files and print on the standard output合并文件并输出 主要用法 1.cat f1.txt,查看f1.txt文件的内容. 2.cat ...
- cat命令
[cat] 合并文件和打印到标准输出 命令格式: cat [OPTION]... [FILE]... 命令功能: 拼接文件或者做标准输入输出 命令格式: cat [OPTION].. ...
- 查看cpu的信息cat /proc/cpuinfo
cat /proc/cpuinfo processor : vendor_id : GenuineIntel cpu family : model : model name : Intel(R) Co ...
随机推荐
- Tunnel Warfare(HDU1540+线段树+区间合并)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1540 题目: 题意:总共有n个村庄,有q次操作,每次操作分为摧毁一座村庄,修复一座村庄,和查询与询问的 ...
- this可以通过call改变的测试
- 关于this问题
对于关键字this,其实很好理解,谁调用我就指向谁.下面举个例子说明: 其实这也是在学习闭包中的一个案例: var name = "The window"; var obj = { ...
- GCC在C语言中内嵌汇编 asm __volatile__ 【转】
转自:http://blog.csdn.net/pbymw8iwm/article/details/8227839 在内嵌汇编中,可以将C语言表达式指定为汇编指令的操作数,而且不用去管如何将C语言表达 ...
- python基础===jieba模块,Python 中文分词组件
api参考地址:https://github.com/fxsjy/jieba/blob/master/README.md 安装自行百度 基本用法: import jieba #全模式 word = j ...
- C json实战引擎 二 , 实现构造部分
引言 这篇博文和前一篇 C json实战引擎一,实现解析部分设计是相同的,都是采用递归下降分析. 这里扯一点 假如你是学生 推荐一本书 给 大家 自制编程语言 http://baike.baidu.c ...
- 12-5 NSSet
原文:http://rypress.com/tutorials/objective-c/data-types/nsset NSSet NSSet, NSArray, and NSDictionary ...
- Redis Cluster 集群使用(3)
简介 Redis3.0版本之前,可以通过Redis Sentinel(哨兵)来实现高可用(HA),从3.0版本之后,官方推出了Redis Cluster,它的主要用途是实现数据分片(Data Shar ...
- apusic7配置2
1:<SERVICE class="com.apusic.web.WebService" > <ATTRIBUTE NAME="MaxWaitingCl ...
- P1968
题目背景 此处省略maxint+1个数 题目描述 在以后的若干天里戴维将学习美元与德国马克的汇率.编写程序帮助戴维何时应买或卖马克或美元,使他从100美元开始,最后能获得最高可能的价值. 输入输出格式 ...