P1600 [NOIP2016 提高组] 天天爱跑步 (树上差分)
对于一条路径,s-t,位于该路径上的观察员能观察到运动员当且仅当以下两种情况成立:(d[ ]表示节点深度)
1.观察员x在s-lca(s,t)上时,满足d[s]=d[x]+w[x]就能观察到,所以我们在这条路径上每个点都放置一个d[s]的物品(差分实现),所有路径处理完后dfs一遍,查询每个节点d[x]+w[x]的物品有多少个就是该种情况的答案。
2.观察员x在lca(s,t)-t上时,同理有d[s]+d[t]-2*d[z]-w[x]=d[t]-d[x],移项的d[s]-2*d[z]=w[x]-d[x],放置d[s]-2*d[z]的物品,查询w[x]-d[x]的物品有多少个,(注意w[x]-d[x]有可能是负数,所以都加上一个n)。
代码中用vector存每个节点的物品,a1,a2是第一种情况的+1和-1;b1,b2是第二种情况的+1和-1;
对于查询子树和,开两个桶实现(全局开桶),满足区间减法,递归回溯回来后-刚遍历时就是子树和,也就是答案。
1 #include<iostream>
2 #include<cstdio>
3 #include<cstring>
4 #include<algorithm>
5 #include<cmath>
6 #include<vector>
7 #include<queue>
8 using namespace std;
9 const int N=300010;
10 int to[N<<1],nxt[N<<1],head[N],tot;
11 int f[N][20],d[N],w[N],v[N];
12 int c1[N<<1],c2[N<<1],ans[N];
13 int n,m,t;
14 queue<int> q;
15 vector<int> a1[N],b1[N],a2[N],b2[N];
16
17 void add(int x,int y){
18 nxt[++tot]=head[x];
19 head[x]=tot;
20 to[tot]=y;
21 }
22
23 void bfs(){
24 t=log(n)/log(2);
25 q.push(1);d[1]=1;
26 while(q.size()){
27 int x=q.front();q.pop();
28 for(int i=head[x];i;i=nxt[i]){
29 int y=to[i];
30 if(d[y]) continue;
31 d[y]=d[x]+1;
32 f[y][0]=x;
33 for(int j=1;j<=t;j++)
34 f[y][j]=f[f[y][j-1]][j-1];
35 q.push(y);
36 }
37 }
38 }
39
40 int lca(int x,int y){
41 if(d[x]>d[y]) swap(x,y);
42 for(int i=t;i>=0;i--)
43 if(d[f[y][i]]>=d[x]) y=f[y][i];
44 if(x==y) return x;
45 for(int i=t;i>=0;i--)
46 if(f[x][i]!=f[y][i]) x=f[x][i],y=f[y][i];
47 return f[x][0];
48 }
49
50 void dfs(int x){
51 int val1=c1[d[x]+w[x]],val2=c2[w[x]-d[x]+n];//防止负数,加一个n
52 v[x]=1;
53 for(int i=head[x];i;i=nxt[i]){
54 int y=to[i];
55 if(v[y]) continue;
56 dfs(y);
57 }
58 for(int i=0;i<a1[x].size();i++) c1[a1[x][i]]++;
59 for(int i=0;i<b1[x].size();i++) c1[b1[x][i]]--;
60 for(int i=0;i<a2[x].size();i++) c2[a2[x][i]+n]++;
61 for(int i=0;i<b2[x].size();i++) c2[b2[x][i]+n]--;
62 ans[x]+=c1[d[x]+w[x]]-val1+c2[w[x]-d[x]+n]-val2;//末态-初态,相当于就是统计了子树和
63 }
64
65 int main(){
66 cin>>n>>m;
67 for(int i=1;i<n;i++){
68 int x,y;
69 scanf("%d%d",&x,&y);
70 add(x,y);add(y,x);
71 }
72 for(int i=1;i<=n;i++) scanf("%d",&w[i]);
73 bfs();
74 for(int i=1;i<=m;i++){
75 int x,y;scanf("%d%d",&x,&y);
76 int z=lca(x,y);
77 a1[x].push_back(d[x]);
78 b1[f[z][0]].push_back(d[x]);
79 a2[y].push_back(d[x]-2*d[z]);
80 b2[z].push_back(d[x]-2*d[z]);
81 }
82 dfs(1);
83 for(int i=1;i<n;i++) printf("%d ",ans[i]);
84 printf("%d\n",ans[n]);
85 return 0;
86 }
P1600 [NOIP2016 提高组] 天天爱跑步 (树上差分)的更多相关文章
- [NOIp2016提高组]天天爱跑步
题目大意: 有一棵n个点的树,每个点上有一个摄像头会在第w[i]秒拍照. 有m个人再树上跑,第i个人沿着s[i]到t[i]的路径跑,每秒钟跑一条边. 跑到t[i]的下一秒,人就会消失. 问每个摄像头会 ...
- NOIP2016 Day1 T2 天天爱跑步(树上差分,LCA)
原文链接 原题链接 题目描述 小c同学认为跑步非常有趣,于是决定制作一款叫做<天天爱跑步>的游戏.<天天爱跑步>是一个养成类游戏,需要玩家每天按时上线,完成打卡任务. 这个游戏 ...
- Luogu P1600[NOIP2016]day1 T2天天爱跑步
号称是noip2016最恶心的题 基本上用了一天来搞明白+给sy讲明白(可能还没讲明白 具体思路是真的不想写了(快吐了 如果要看,参见洛谷P1600 天天爱跑步--题解 虽然这样不好但我真的不想写了 ...
- 洛谷 1600 (NOIp2016) 天天爱跑步——树上差分
题目:https://www.luogu.org/problemnew/show/P1600 看TJ:https://blog.csdn.net/clove_unique/article/detail ...
- 洛谷P1600 天天爱跑步——树上差分
题目:https://www.luogu.org/problemnew/show/P1600 看博客:https://blog.csdn.net/clove_unique/article/detail ...
- 洛谷$P1600$ 天天爱跑步 树上差分
正解:树上差分 解题报告: 传送门$QwQ$! 这题还挺妙的,,,我想了半天才会$kk$ 首先对一条链$S-T$,考虑先将它拆成$S-LCA$和$LCA-T$,分别做.因为总体上来说差不多接下来我就只 ...
- NOIP2016 天天爱跑步 (树上差分+dfs)
题目大意:给你一颗树,树上每个点都有一个观察员,他们仅会在 w[i] 时刻出现,观察正在跑步的玩家 一共有m个玩家,他们分别从节点 s[i] 同时出发,以每秒跑一条边的速度,沿着到 t[i] 的唯一路 ...
- [NOIP2016]天天爱跑步(树上差分+线段树合并)
将每个人跑步的路径拆分成x->lca,lca->y两条路径分别考虑: 对于在点i的观察点,这个人(s->t)能被观察到的充要条件为: 1.直向上的路径:w[i]=dep[s]-dep ...
- NOIP2016 天天爱跑步 - 树上差分
传送门 题目分析: 一年前还是个傻子的时候居然直接放弃了这题. 首先列出两个方程:如果i节点的观察员能够观察到由s->t的那个人,那么: \[dep[s] - dep[i] = w[i], de ...
随机推荐
- 【水】关于 __attribute__
目录 1. weak 2. aligned 3. __packed__ 4. always_inline 1. weak 实现如果 Func1 被定义了,我就调用 Func1, 否则就调用 MyFun ...
- 从RabbitMQ平滑迁移到RocketMQ技术实战
作者:vivo 互联网中间件团队- Liu Runyun 大量业务使用消息中间件进行系统间的解耦.异步化.削峰填谷设计实现.公司内部前期基于RabbitMQ实现了一套高可用的消息中间件平台.随着业务的 ...
- PROFIBUS,PROFINET,Ethernet三者的区别
PROFIBUS,PROFINET,Ethernet三者的区别 简单一点说,PROFINET=PROFIbus+etherNET,把Profibus的主从结构移植到以太网上,所以profinet会有C ...
- GCC常见命令
rwx 对于目录和文件的区别 文件 目录 r 文件的内容可以被查看.支持cat.more.head...vim 目录的内容可以被查看.ls.tree w 文件的内容可以被添加.修改.删除.vim &g ...
- BZOJ4337 树的同构 (树哈希)(未完成)
样例迷,没过 交了30pts #include <cstdio> #include <iostream> #include <cstring> #include & ...
- 10大python加速技巧
简介 目前非常多的数据竞赛都是提交代码的竞赛,而且加入了时间的限制,这就对于我们python代码的加速非常重要.本篇文章我们介绍在Python中加速代码的一些技巧.可能不是很多,但在一些大的循环或者函 ...
- 业界压测平台与JMeter的对比
压测平台是什么? 压测,即压力测试,作用是对各种服务对象进行压力测试以获得该服务处于或超过预期负载时系统的运行情况,进而判断系统在峰值负载或超出最大负载情况下的处理能力. 压测工具,顾名思义,就是用来 ...
- RAID磁盘阵列技术
RAID磁盘阵列技术 1.RAID概述 RAID(Redundant Array of Independent Disk),从字面意思讲的是基于独立磁盘的具有冗余的磁盘阵列,其核心思想是将多块独立磁盘 ...
- java-运算符以及简单运用
运算符: 1)赋值运算符:= 2)算术运算符:+-*/%,++,-- 3)关系运算符:>,<,>=,<=,==,!= boolean 4)逻辑运算符:&&,|| ...
- JedisConnectionException: java.net.SocketException: Broken pipe (Write failed) 问题排查
问题描述 笔者有2个应用会不定时请求redis,其中一个应用大约每分钟请求一次,可以正常请求,但是另一个大约每小时请求一次的应用,经常出现Broken pipe (Write failed)报错,具体 ...