原题链接

题意简述

给出一棵n(n≤105)个节点的树,每个点有点权。每次可以选择两个叶节点并将连接它们的路径上的节点的点权-1(包括叶节点)。求能否将所有节点的点权都变为0。

分析



先考虑最简单的情况。在这种情况下,au必须等于av,否则GG。因为要想对v操作只能通过u,想对u操作只能通过v。

若相等我们可以令bu=av,并定义bu为:u需要往外连bu条路径。因为需要有au条路径进到以u为根的子树里面,可以看做u需要向外连au条路径。



再考虑一般情况。在这种情况下,au必须小于等于∑bv,否则GG。因为即使把bv都减完了au也不能为0,并且已经没有办法再减少au了。

若au=∑bv,直接从u往外连au条路径就好了。

和刚才不同,v之间可以自行解决一部分。比如操作v1−u−v2,可以让bu减1,让∑bv减2。我们可以进行类似的操作直到bu=∑bv,然后同上。

但是有可能没法让bu=∑bv,那就GG。那什么情况下不可行呢?

结论:当max{bv}≤(∑bv)/2时,可以把所有的bv消成0(或者剩一个1)。

证明

可能不对,看看就好

我们可以把问题反转一下:

对于一个零序列,每次对两个位置+1,能否得到目标序列?

显然的结论有当 max{bv}>(∑bv)/2 时会有 max{bv}−(∑bv−max{bv}) 加不出来。以及当 ∑bv 为奇数时至少会剩下一个。

下证对于其他情况:

额外创建两个位置,对它们进行无限次操作,意思就是足够多次。

{0,0,...,0} -> {0,0,...,0(,inf,inf)}

这时候我们加入了一种新操作:令这两个inf减1,也就是撤回一次。 然后我们可以做到 :

{0,0,...,0(,inf,inf)} -> {1,0,...,0(,inf+1,inf)} -> {2,0,...,0(,inf+1,inf+1)} -> {2,0,...,0(,inf,inf)}。

这样就有了构造方法:先两两给所有奇数填上1(要是有奇数个奇数就说明∑bv为奇数肯定会剩下,所以可以把一个奇数视为偶数),然后通过以上+2的操作把所有数都填好。最后对额外的两个位置一直-1减到0,这样就构造完成了。

如果在任何时候出现bu无法等于∑bv,那么GG。

以及,broot≠0也GG。

遍历所有节点复杂度为O(n),遍历每个节点的所有子节点复杂度为O(n),总时间复杂度为O(n)。

实现

首先以一个度不为1的点作为root,然后DFS出深度dpt和树的结构

由下到上将节点u和它的子节点v合并出bu,最后检查 broot

代码

//Cleaning
#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std;
typedef long long lint;
int const N=1e5+10;
int n,a[N];
int cnt,h[N],deg[N];
struct edge{
int u,v,nxt;
edge(int u1=0,int v1=0)
{
u=u1,v=v1;
nxt=h[u];
h[u]=cnt;
}
}ed[N<<1];
int root,fa[N],dpt[N];
struct rec{int dpt,id;} r[N];
bool cmpDpt(rec x,rec y) {return x.dpt>y.dpt;}
void dfs(int u)
{
for(int i=h[u];i!=0;i=ed[i].nxt)
{
int v=ed[i].v;
if(v==fa[u]) continue;
fa[v]=u,dpt[v]=dpt[u]+1;
dfs(v);
}
}
int main()
{
freopen("c.in","r",stdin);
scanf("%d",&n);
for(int i=1;i<=n;i++) scanf("%d",&a[i]);
if(n==2)
{
if(a[1]!=a[2]) printf("NO");
else printf("YES");
return 0;
}
cnt=0; memset(h,0,sizeof h);
for(int i=1;i<=n-1;i++)
{
int u,v;
scanf("%d%d",&u,&v);
ed[++cnt]=edge(u,v); deg[v]++;
ed[++cnt]=edge(v,u); deg[u]++;
}
for(int i=1;i<=n;i++)
if(deg[i]>1)
{
fa[i]=0,dpt[i]=1;
dfs(root=i);
break;
}
for(int i=1;i<=n;i++) r[i].dpt=dpt[i],r[i].id=i;
sort(r+1,r+n+1,cmpDpt);
for(int i=1;i<=n;i++)
{
int u=r[i].id;
if(deg[u]==1) continue;
int maxx=0; lint sum=0;
for(int j=h[u];j!=0;j=ed[j].nxt)
{
int v=ed[j].v;
if(fa[v]!=u) continue;
maxx=max(maxx,a[v]); sum+=a[v];
}
lint in=min(sum/2,sum-maxx);
if(a[u]>sum) {printf("NO"); return 0;}
else
{
if(sum-a[u]>in) {printf("NO"); return 0;}
else a[u]-=sum-a[u];
}
}
if(a[root]==0) printf("YES");
else printf("NO");
return 0;
}

注意

n=2时要特判一下,因为两个点都是叶节点会找不出root

AGC010 - C: Cleaning的更多相关文章

  1. 【AGC010 C】Cleaning

    题意 有一棵 \(n\) 个点的树,第 \(i\) 个节点有 \(a_i\) 个石子. 每次都可以选择一对不同的叶子节点,这对叶子节点路径上的所有点都必须要有石子.然后去掉这两个叶子节点路径上的每个节 ...

  2. Atcoder Grand Contest 010 C - Cleaning 树贪心(伪)

    C - Cleaning 题目连接: http://agc010.contest.atcoder.jp/tasks/agc010_c Description There is a tree with ...

  3. 【AtCoder】AGC010

    AGC010 A - Addition 如果所有数加起来是偶数那么一定可以,否则不行 #include <bits/stdc++.h> #define fi first #define s ...

  4. 【bzoj1672】[USACO2005 Dec]Cleaning Shifts 清理牛棚

    题目描述 Farmer John's cows, pampered since birth, have reached new heights of fastidiousness. They now ...

  5. Coursera-Getting and Cleaning Data-week1-课程笔记

    博客总目录,记录学习R与数据分析的一切:http://www.cnblogs.com/weibaar/p/4507801.html -- Sunday, January 11, 2015 课程概述 G ...

  6. Coursera-Getting and Cleaning Data-Week2-课程笔记

    Coursera-Getting and Cleaning Data-Week2 Saturday, January 17, 2015 课程概述 week2主要是介绍从各个来源读取数据.包括MySql ...

  7. Coursera-Getting and Cleaning Data-Week3-dplyr+tidyr+lubridate的组合拳

    Coursera-Getting and Cleaning Data-Week3 Wednesday, February 04, 2015 好久不写笔记了,年底略忙.. Getting and Cle ...

  8. Coursera-Getting and Cleaning Data-week4-R语言中的正则表达式以及文本处理

    博客总目录:http://www.cnblogs.com/weibaar/p/4507801.html Thursday, January 29, 2015 补上第四周笔记,以及本次课程总结. 第四周 ...

  9. 【BZOJ1672】[Usaco2005 Dec]Cleaning Shifts 清理牛棚 动态规划

    [BZOJ1672][Usaco2005 Dec]Cleaning Shifts Description Farmer John's cows, pampered since birth, have ...

随机推荐

  1. Servlet--HttpServletResponse的2个操作流的方法

    前面已经说过无数多次了,我们的项目都是基于HTTP协议的一次请求,一次响应.实际编码中,我们在处理完逻辑后一般是跳转到一个页面上,或者用输出流返回json字符串.其实跳转到一个页面往往也就是JSP,J ...

  2. 对datatable操作经验-排序和分页

    1.datatable排序1: public DataTable SortDesc(DataTable dt){ DataView dv = new DataView(); dv.Table = dt ...

  3. 重置CentOS 7的Root密码

    centos7与centos6有很多修改,不一样了,打算写几篇关于日常用到的改动 修改root密码 centos7的用户模式跟6有所不同 1 - 在启动grub菜单,选择编辑选项启动 2 - 按键盘e ...

  4. NIO笔记---上

    小弟前端时间由于开发个管理系统导致断更了近20天!!马上就要春招了,学习了一下NIO,将笔记记录下,希望和我一样的18届毕业生都能找到满意的公司!! 本文记录了NIO与IO的区别,缓冲区的数据存取,直 ...

  5. python并发编程之多进程(实现)

    一.multipricessing模块的介绍 python中的多线程无法利用多核优势,如果想要充分的使用多核CPU资源,在python中大部分情况下需要用多线程,python提供了multiproce ...

  6. 使用命令行生成jar包

    测试用类 public class Hello { public static void main(String[] args) { System.out.println("hello wo ...

  7. JAVA并发编程学习笔记------FutureTask

    FutureTask是Future和Callable的结合体.传统的代码是这样写的Future f = executor.submit(new Callable()); 然后通过Future来取得计算 ...

  8. ABAP系统值

    SY-SUBRC:语句执行后的返回值,0表示成功 SY-DATUM:当前服务器日期 SY-UZEIT:当前服务器时间 SY-ULINE:255长度的水平线 SY-VLINE:垂直线 SY-INDEX: ...

  9. xBIM 格式之间转换

    目录 xBIM 应用与学习 (一) xBIM 应用与学习 (二) xBIM 基本的模型操作 xBIM 日志操作 XBIM 3D 墙壁案例 xBIM 格式之间转换 xBIM 使用Linq 来优化查询 x ...

  10. python 闯关之路一(语法基础)

    1,什么是编程?为什么要编程? 答:编程是个动词,编程就等于写代码,那么写代码是为了什么呢?也就是为什么要编程呢,肯定是为了让计算机帮我们搞事情,代码就是计算机能理解的语言. 2,编程语言进化史是什么 ...