AT2304 Cleaning

AT2304 Cleaning

题意

一个树上每个节点有一些石子,每次只能选取两个叶子节点并将路径间的所有点上的石子数量减1,问是否能将所有石子取完。

思路

fx 表示从 x 节点向上的路径条数,sx 为子节点的 f 值的和,则有:

ax=sxfx2+fxfx=2axsx

我们只需要保证以下条件即可:

  • 从子节点传上来的路径条数的最大值小于等于该点石头个数;
  • 向上传的路径条数不为负且小于等于该点石头数。

也就是说,在合法条件下,我们令能在子树内匹配的路径数尽量多,然后向上传路径。

可以证明,在满足上面两个条件下,总能构造出一种合法方案使这个点合法。

其他条件:

  • 子叶节点的 fx=ax

  • 根节点不能为子节点;

  • 若根节点的 f 值不为0,判定为无解。

实现

#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cctype>
#include<cstring>
#include<cmath>
using namespace std;
inline int read(){
	int w=0,x=0;char c=getchar();
	while(!isdigit(c))w|=c=='-',c=getchar();
	while(isdigit(c))x=x*10+(c^48),c=getchar();
	return w?-x:x;
}
namespace star
{
	const int maxn=1e5+10;
	int n,a[maxn],root;
	int ecnt,head[maxn],to[maxn<<1],nxt[maxn<<1],in[maxn],f[maxn];
	inline void addedge(int a,int b){
		to[++ecnt]=b,nxt[ecnt]=head[a],head[a]=ecnt;in[a]++;
		to[++ecnt]=a,nxt[ecnt]=head[b],head[b]=ecnt;in[b]++;
	}
	void dfs(int x,int fa){
		f[x]=in[x]==1?a[x]:(a[x]<<1);
		for(int i=head[x];i;i=nxt[i]){
			int u=to[i];
			if(u==fa)continue;
			dfs(u,x);
			f[x]-=f[u];
			if(f[u]>a[x]) puts("NO"),exit(0);
		}
		if(f[x]>a[x] or f[x]<0) puts("NO"),exit(0);
	}
	inline void work(){
		n=read();
		for(int i=1;i<=n;i++) a[i]=read();
		for(int i=1;i<n;i++) addedge(read(),read());
		if(n==2) return puts(a[1]==a[2]?"YES":"NO"),void();
		for(int i=1;i<=n;i++) if(in[i]!=1){root=i;break;}
		dfs(root,0);
		puts(f[root]?"NO":"YES");
	}
}
signed main(){
	star::work();
	return 0;
}
posted @   Star_Cried  阅读(86)  评论(0编辑  收藏  举报
编辑推荐:
· go语言实现终端里的倒计时
· 如何编写易于单元测试的代码
· 10年+ .NET Coder 心语,封装的思维:从隐藏、稳定开始理解其本质意义
· .NET Core 中如何实现缓存的预热?
· 从 HTTP 原因短语缺失研究 HTTP/2 和 HTTP/3 的设计差异
阅读排行:
· 分享一个免费、快速、无限量使用的满血 DeepSeek R1 模型,支持深度思考和联网搜索!
· 使用C#创建一个MCP客户端
· ollama系列1:轻松3步本地部署deepseek,普通电脑可用
· 基于 Docker 搭建 FRP 内网穿透开源项目(很简单哒)
· 按钮权限的设计及实现
点击右上角即可分享
微信分享提示