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 ...
随机推荐
- Mybatis源码解读-配置加载和Mapper的生成
问题 Mybatis四大对象的创建顺序? Mybatis插件的执行顺序? 工程创建 环境:Mybatis(3.5.9) mybatis-demo,参考官方文档 简单示例 这里只放出main方法的示例, ...
- vue-resource && axios
1 # axios 2 # 1.安装:npm i axios 3 # 2.使用: 4 import axios from 'axios' 5 axios.get(URL).then(response= ...
- SAM复杂度证明
关于$SAM$的复杂度证明(大部分是对博客的我自己的理解和看法) 这部分是我的回忆,可省略 先回忆一下$SAM$ 我所理解的$SAM$,首先扒一张图 初始串$aabbabd$ 首先发现,下图里的$S- ...
- Luogu1038 神经网络 (拓扑排序)
拓扑排序,裸的,水的. 第一发:题读错,输出错,输入错,到处错 \(\longrightarrow\) 40pts (excuse me ?) 第二发:漏了输入层特判 \(\longrightarro ...
- Android 自动取色并设置沉浸式状态栏
Android 自动取色并设置沉浸式状态栏 - Stars-One的杂货小窝 最近在进行产品的优化,也是研究了下沉浸式状态栏的实现方法及自动取色,记录一下笔记 设置沉浸式状态栏 1.添加依赖 这里,是 ...
- Merge Into 语法支持
KINGBASE 兼容Oracle 语法,实现了merge into 的功能.以下以例子的形式,介绍merge into语法的使用.以下例子在V8R6 ,且 database_mode=oracle ...
- KingbaseES中的参数查看与修改
KingbaseES数据库的配置参数都在kingbase.conf文件中,这些参数有些是直接修改就可以生效,有些需要重启数据库才能生效,而有些根本就不能修改.数据库把这些参数分为以下几类: inter ...
- IIS 实现http重定向https(亲测有效:解决URL重写模块配置https重定向不生效的问题)
前言 以前部署网站的时候,都是通过代码来实现http重定向https,最近在部署个人网站的时候,突发奇想可不可通过IIS来实现无代码的重定向呢? 在一番操作猛如虎的搜索引擎操作后,发现只有google ...
- 【读书笔记】C#高级编程 第十五章 反射
(一)在运行期间处理和检查代码 自定义特性允许把自定义元数据与程序元素关联起来.反射是一个普通术语,它描述了在运行过程中检查和处理程序元素的功能.例如,反射允许完成的任务: 枚举类型的成员 实例化新对 ...
- salesforce零基础学习(一百一十七)salesforce部署方式及适用场景
本篇参考:https://architect.salesforce.com/decision-guides/migrate-change https://developer.salesforce.co ...