【ARC063E】Integers on a tree
Description
给定一棵\(n\)个点的树,其中若干个点的权值已经给出。现在请为剩余点填入一个值,使得相邻两个点的差的绝对值恰好为1。请判断能否实现,如果能,请将方案一并输出。
Solution
卡了一会,终于想出来了。
首先从深度奇偶性和权值奇偶性这一方面考虑:如果所有已知点的权值与深度的奇偶性关系不全一样,则一定无解。
然后考虑怎么构造。如果用已填点将树分成若干块,显然每一块是独立的,现在考虑单独一块。
直接想有一点困难,所以我们先尝试考虑每一个空点\(u\)能填什么数:考虑这一个块中的一个有值点,其权值为\(x\),如果它到\(u\)距离为\(2k\),那么\(u\)的填数选择就有\(x-2k,x-2k+2,...,x,...,x+2k-1,x+2k\);如果它到\(u\)距离为\(2k+1\),那么\(u\)的填数选择就有\(x-2k-1,x-2k+1,x-2k+3...,x+2k-1,x+2k+1\)。
考虑所有的有值点,那么\(u\)的取值范围就是这些选择的交\(S\)。也就是说,只要\(u\)填\(S\)中的权值,单看\(u\)而言就一定能填出满足所有有值点的方案。若\(S\)为空则全局无解。
可是对于每个空点,我们到底选\(S\)中的哪个权值填入呢?注意到如果随便填的话,可能会出现跳跃的问题。
我画了几个例子。构造例子的方法是先弄一棵填好权值的合法树,再指定有值位置。当我用上述方法考虑空点的\(S\)时,我们发现:当且仅当将每一个点都取其\(S\)中的最小值时有解(或都取最大值),原问题才有解,这种填法即一种合法方案;否则无解。
粗略证明:考虑一个点\(u\)的取值集合\(S\),它其实是一个范围\([l,r]\),但中间的取值是每隔1取一个的。对于任意一个与\(u\)相邻的点\(v\),记其权值范围为\([l',r']\),则其权值边界的跨度都不会超过1,即有\(l'=l\pm1\)和\(r'=r \pm1\),注意两者不是互不相关的。为什么?\(S\)记录的是每一个有值点\(x\)到这个点对应的权值范围的交。走多一步,意味着空隙翻转(原来是跳一次取一次的),对于走近了的\(x\),其权值范围以\(x\)为中心向内空隙翻转,对于走远了的\(x\),其权值范围向外空隙翻转;也就是一个多了两端,一个少了两端。仔细分析下来,\(S\)的边界变化也不会超过1.
如果有解,那么这样填数一定能够满足条件------我们是贴着边界走的,而有值点本身也在边界上。如果这样填都不能满足,显然全局无解。
因此我们对每个点取\(S\)的最小值,判断是否合法即可。
至于\(S\)最小值的计算方法,这里有一个技巧:对于每个点\(u\),我们直接维护所有有值点\(x\)对应的范围的左端点的最大值,即\(x-dis\)的最大值。这样一来,如果真正意义上\(S\)交集不为空,那么这个值就是\(u\)的取值。否则,这个值无论如何都会使得后面的判定出错不合法,毕竟取值不满足所有的有值点。
Code
#include <cstdio>
#include <cstring>
using namespace std;
const int N=100005;
const int INF=1e9;
int n,m;
int a[N];
int h[N],tot,dep[N];
struct Edge{
int v,next;
}e[N*2];
int f[N],g[N],ans[N];
inline int max(int x,int y){
return x>y?x:y;
}
inline int abs(int x){
return x>=0?x:-x;
}
void addEdge(int u,int v){
e[++tot]=(Edge){v,h[u]}; h[u]=tot;
e[++tot]=(Edge){u,h[v]}; h[v]=tot;
}
void readData(){
scanf("%d",&n);
int u,v;
for(int i=1;i<n;i++){
scanf("%d%d",&u,&v);
addEdge(u,v);
}
memset(a,-1,sizeof a);
scanf("%d",&m);
for(int i=1;i<=m;i++){
scanf("%d%d",&u,&v);
a[u]=v;
}
}
void mark_dfs(int u,int fa){
dep[u]=dep[fa]+1;
for(int i=h[u],v;i;i=e[i].next)
if((v=e[i].v)!=fa)
mark_dfs(v,u);
}
bool firstCheck(){
mark_dfs(1,0);
int flag=-1;
for(int u=1;u<=n;u++)
if(a[u]!=-1){
if(flag==-1)
flag=(a[u]^dep[u])&1;
else if(((a[u]^dep[u])&1)!=flag)
return false;
}
return true;
}
void dp_dfs1(int u,int fa){
f[u]=(a[u]!=-1)?a[u]:-INF;
for(int i=h[u],v;i;i=e[i].next)
if((v=e[i].v)!=fa){
dp_dfs1(v,u);
f[u]=max(f[u],f[v]-1);
}
}
bool dp_dfs2(int u,int fa){
if(a[u]!=-1)
g[u]=a[u];
ans[u]=(a[u]!=-1)?a[u]:max(f[u],g[u]);
if(fa&&abs(ans[u]-ans[fa])!=1)
return false;
static int ch[N],cnt;
static int l[N],r[N];
cnt=0;
for(int i=h[u],v;i;i=e[i].next)
if((v=e[i].v)!=fa)
ch[++cnt]=v;
l[0]=g[u]+1; r[cnt+1]=-INF;
for(int i=1;i<=cnt;i++)
l[i]=max(l[i-1],f[ch[i]]);
for(int i=cnt;i>=1;i--)
r[i]=max(r[i+1],f[ch[i]]);
for(int i=1;i<=cnt;i++)
g[ch[i]]=max(l[i-1],r[i+1])-2;
for(int i=h[u],v;i;i=e[i].next)
if((v=e[i].v)!=fa){
if(dp_dfs2(v,u)==false)
return false;
if(abs(ans[u]-ans[v])!=1)
return false;
}
return true;
}
bool solve(){
dp_dfs1(1,0);
g[1]=-INF;
if(dp_dfs2(1,0)==false)
return false;
puts("Yes");
for(int i=1;i<=n;i++)
printf("%d\n",ans[i]);
return true;
}
int main(){
readData();
if(!firstCheck()||!solve()){
puts("No");
return 0;
}
return 0;
}
【ARC063E】Integers on a tree的更多相关文章
- 【BZOJ2959】长跑(Link-Cut Tree,并查集)
[BZOJ2959]长跑(Link-Cut Tree,并查集) 题面 BZOJ 题解 如果保证不出现环的话 妥妥的\(LCT\)傻逼题 现在可能会出现环 环有什么影响? 那就可以沿着环把所有点全部走一 ...
- 【BZOJ2588】Count On a Tree(主席树)
[BZOJ2588]Count On a Tree(主席树) 题面 题目描述 给定一棵N个节点的树,每个点有一个权值,对于M个询问(u,v,k),你需要回答u xor lastans和v这两个节点间第 ...
- 【BZOJ2816】【ZJOI2012】网络(Link-Cut Tree)
[BZOJ2816][ZJOI2012]网络(Link-Cut Tree) 题面 题目描述 有一个无向图G,每个点有个权值,每条边有一个颜色.这个无向图满足以下两个条件: 对于任意节点连出去的边中,相 ...
- 【CF434E】Furukawa Nagisa's Tree 点分治
[CF434E]Furukawa Nagisa's Tree 题意:一棵n个点的树,点有点权.定义$G(a,b)$表示:我们将树上从a走到b经过的点都拿出来,设这些点的点权分别为$z_0,z_1... ...
- 【CF725G】Messages on a Tree 树链剖分+线段树
[CF725G]Messages on a Tree 题意:给你一棵n+1个节点的树,0号节点是树根,在编号为1到n的节点上各有一只跳蚤,0号节点是跳蚤国王.现在一些跳蚤要给跳蚤国王发信息.具体的信息 ...
- 【SPOJ】QTREE7(Link-Cut Tree)
[SPOJ]QTREE7(Link-Cut Tree) 题面 洛谷 Vjudge 题解 和QTREE6的本质是一样的:维护同色联通块 那么,QTREE6同理,对于两种颜色分别维护一棵\(LCT\) 每 ...
- 【SPOJ】Count On A Tree II(树上莫队)
[SPOJ]Count On A Tree II(树上莫队) 题面 洛谷 Vjudge 洛谷上有翻译啦 题解 如果不在树上就是一个很裸很裸的莫队 现在在树上,就是一个很裸很裸的树上莫队啦. #incl ...
- 【BZOJ2870】最长道路tree 点分治+树状数组
[BZOJ2870]最长道路tree Description H城很大,有N个路口(从1到N编号),路口之间有N-1边,使得任意两个路口都能互相到达,这些道路的长度我们视作一样.每个路口都有很多车辆来 ...
- 【数据挖掘】分类之decision tree(转载)
[数据挖掘]分类之decision tree. 1. ID3 算法 ID3 算法是一种典型的决策树(decision tree)算法,C4.5, CART都是在其基础上发展而来.决策树的叶子节点表示类 ...
随机推荐
- (译)理解 LSTM 网络 (Understanding LSTM Networks by colah)
@翻译:huangyongye 原文链接: Understanding LSTM Networks 前言:其实之前就已经用过 LSTM 了,是在深度学习框架 keras 上直接用的,但是到现在对LST ...
- sklearn学习笔记之简单线性回归
简单线性回归 线性回归是数据挖掘中的基础算法之一,从某种意义上来说,在学习函数的时候已经开始接触线性回归了,只不过那时候并没有涉及到误差项.线性回归的思想其实就是解一组方程,得到回归函数,不过在出现误 ...
- 【php增删改查实例】第十九节 - session的使用: 让服务器知道你是谁?
因为HTTP请求是一种无状态的请求,所谓无状态,就是服务器不会记录下你本次请求的信息.http它是基于请求 - 相应模式的一种数据传输协议.就是说,你发送一个请求,我服务器给你一个响应,这件事情就算完 ...
- Linux系统安装IDS(snort工具)
第一步:预装daq所需程序 snort使用数据采集器(daq)监听防火墙数据包队列,所以按照daq.需预装的程序有:flex.bison.libcap. sudo apt-get install fl ...
- 【nodejs】让nodejs像后端mvc框架(asp.net mvc)一orm篇【如EF般丝滑】typeorm介绍(8/8)
文章目录 前情概要 在使用nodejs开发过程中,刚好碰到需要做一个小工具,需要用到数据库存储功能.而我又比较懒,一个小功能不想搞一个nodejs项目,又搞一个后端项目.不如直接在nodejs里面把对 ...
- Microsoft Office软件自定义安装目录
Microsoft Office安装时不能手动设置安装目录,本文描述通过修改注册表的方式自定义安装目录 1.同时按下快捷键 win + r 启动运行 2.输入 regedit 打开注册表 3.找到 ...
- SC1243sensor噪点问题调试
接手一块SC1243sensor的板子调试,仔细核对了原理图和PCB发现,PCB不是很好,电源处理不够好,但是出图了,问题是有噪点,麻点,根据经验要求软件修改了PCLK的极性噪点消失,问题解决. 1: ...
- Swarm基于多主机容器网络 - overlay networks 梳理
前面介绍了Docker管理工具-Swarm部署记录,下面重点说下Swarm基于多主机容器通信的覆盖网络 在Docker版本1.12之后swarm模式原生支持覆盖网络(overlay networks) ...
- [Android]记录一次处理app:transformDexArchiveWithExternalLibsDexMergerForDebug错误
第一种情况: Android 目录结构如下: app中build.gradle包含: implementation 'com.squareup.okhttp3:okhttp:3.6.0' implem ...
- 1013 B. And
链接 [http://codeforces.com/contest/1013/problem/B] 题意 给你一个n和x,再给n个数,有一种操作用x&a[i]取代,a[i],问使其中至少两个数 ...