【NOIP2016】天天爱跑步
题目描述
小c同学认为跑步非常有趣,于是决定制作一款叫做《天天爱跑步》的游戏。«天天爱跑步»是一个养成类游戏,需要玩家每天按时上线,完成打卡任务。
这个游戏的地图可以看作一一棵包含 个结点和 条边的树, 每条边连接两个结点,且任意两个结点存在一条路径互相可达。树上结点编号为从到的连续正整数。
现在有个玩家,第个玩家的起点为 ,终点为 。每天打卡任务开始时,所有玩家在第秒同时从自己的起点出发, 以每秒跑一条边的速度, 不间断地沿着最短路径向着自己的终点跑去, 跑到终点后该玩家就算完成了打卡任务。 (由于地图是一棵树, 所以每个人的路径是唯一的)
小C想知道游戏的活跃度, 所以在每个结点上都放置了一个观察员。 在结点的观察员会选择在第秒观察玩家, 一个玩家能被这个观察员观察到当且仅当该玩家在第秒也理到达了结点 。 小C想知道每个观察员会观察到多少人?
注意: 我们认为一个玩家到达自己的终点后该玩家就会结束游戏, 他不能等待一 段时间后再被观察员观察到。 即对于把结点作为终点的玩家: 若他在第秒重到达终点,则在结点的观察员不能观察到该玩家;若他正好在第秒到达终点,则在结点的观察员可以观察到这个玩家。
输入输出格式
输入格式:
第一行有两个整数和 。其中代表树的结点数量, 同时也是观察员的数量, 代表玩家的数量。
接下来 行每行两个整数和 ,表示结点 到结点 有一条边。
接下来一行 个整数,其中第个整数为 , 表示结点出现观察员的时间。
接下来 行,每行两个整数,和,表示一个玩家的起点和终点。
对于所有的数据,保证 。
输出格式:
输出1行 个整数,第个整数表示结点的观察员可以观察到多少人。
输入输出样例
6 3
2 3
1 2
1 4
4 5
4 6
0 2 5 1 2 3
1 5
1 3
2 6
2 0 0 1 1 1
5 3
1 2
2 3
2 4
1 5
0 1 0 3 0
3 1
1 4
5 5
1 2 1 0 1
说明
【样例1说明】
对于1号点,,故只有起点为1号点的玩家才会被观察到,所以玩家1和玩家2被观察到,共有2人被观察到。
对于2号点,没有玩家在第2秒时在此结点,共0人被观察到。
对于3号点,没有玩家在第5秒时在此结点,共0人被观察到。
对于4号点,玩家1被观察到,共1人被观察到。
对于5号点,玩家1被观察到,共1人被观察到。
对于6号点,玩家3被观察到,共1人被观察到。
【子任务】
每个测试点的数据规模及特点如下表所示。 提示: 数据范围的个位上的数字可以帮助判断是哪一种数据类型。
题解:
这是一道比较有趣的树上统计题,并没有什么高级算法,只是思想巧妙:
用的大概是一种差分思想,是用桶实现的.(即t[i]表示深度为i的点的个数)
我们要将跑的过程转化成 计算对一个点有贡献的路径的个数
于是我们可以开始讨论.
我们先看一条路径s-t:
1.先是从下往上(s-lca)跑,如果对一个点i有贡献仅当deep[i]+w[i]=deep[s]时.
且对于每一个点,deep[i]+w[i]是定值.所以我们把所有路径的S打上标记,
然后按照dfs序遍历,如果一个点x有标记就把t[deep[x]]+=mark[x].
然后每一个点都做同样的操作:ans[x]+=t[deep[i]+w[i]].
2.然后是lca-t,如果满足L-deep[t]=w[i]-dep[i]时可产生贡献,同样在终点t打上++标记,
然后再回溯的回程中就会把t-lca有贡献的点都统计完 考虑到等式两边都有可能产生负数,
于是就把数组偏移一个N,即所有的访问和标记都+N
3.考虑到子树之间 和 子树对父节点以上节点 统计的影响 我们要在lca处打上标记,消除对父节点以上节点的影响
然后进入下一层dfs之前要保存t[i]的值,设为pre,那么答案就为回溯以后的t[i]-pre.
4.然后调样例的时候会发现如果lca满足统计条件会被统计两次,于是最后的时候需要减掉重复情况
#include<iostream>
#include<cstdio>
#include<cmath>
#include<cstring>
#include<vector>
using namespace std;
const int N=,P=;
int gi(){
int str=;char ch=getchar();
while(ch>'' || ch<'')ch=getchar();
while(ch>='' && ch<='')str=str*+ch-'',ch=getchar();
return str;
}
vector<int>q1[N],q2[N],q3[N];
int n,m,head[N],w[N],num=,maxdep=;
struct Lin{
int next,to;
}a[N*];
int Head[N],NUM=;
struct Linn{
int next,to,id;
}e[N*];
struct Path{
int s,t,dis,lca;
}q[N];
int fa[N],dep[N];bool vis[N];
int find(int x){return fa[x]==x?x:fa[x]=find(fa[x]);}
void init(int x,int y){a[++num].next=head[x];a[num].to=y;head[x]=num;a[++num].next=head[y];a[num].to=x;head[y]=num;}
void init2(int x,int y,int idd){e[++NUM].next=Head[x];e[NUM].to=y;e[NUM].id=idd;Head[x]=NUM;e[++NUM].next=Head[y];e[NUM].to=x,e[NUM].id=idd;Head[y]=NUM;}
void tarjan(int x,int last)
{
int u,v,lca;
vis[x]=true;
if(dep[x]>maxdep)maxdep=dep[x];
for(int i=head[x];i;i=a[i].next)
{
u=a[i].to;
if(u==last)continue;
dep[u]=dep[x]+;tarjan(u,x);fa[u]=x;
}
for(int i=Head[x];i;i=e[i].next)
{
v=e[i].to;
if(vis[v])
{
lca=find(v);
q[e[i].id].lca=lca;
q[e[i].id].dis=dep[v]+dep[x]-(dep[lca]<<);
}
}
}
int t[N],ans[N],tt[N*],mark[N];
void dfs1(int x,int last)
{
int u,now=dep[x]+w[x],pre=;
if(now<=maxdep)pre=t[now];
for(int i=head[x];i;i=a[i].next){
u=a[i].to;
if(u==last)continue;
dfs1(u,x);
}
t[dep[x]]+=mark[x];
if(now<=maxdep)ans[x]+=t[now]-pre;
int size=q1[x].size();
for(int i=;i<size;i++)t[q1[x][i]]--;
}
void dfs2(int x,int last)
{
int u,now=w[x]-dep[x]+P,pre=tt[now];
for(int i=head[x];i;i=a[i].next){
u=a[i].to;
if(u==last)continue;
dfs2(u,x);
}
int size=q2[x].size();
for(int i=;i<size;i++)tt[q2[x][i]]++;
ans[x]+=tt[now]-pre;
size=q3[x].size();
for(int i=;i<size;i++)tt[q3[x][i]]--;
}
int main()
{
n=gi();m=gi();
int x,y;
for(int i=;i<n;i++){
x=gi();y=gi();
init(x,y);
}
for(int i=;i<=n;i++)w[i]=gi(),fa[i]=i;
for(int i=;i<=m;i++)
{
q[i].s=gi();q[i].t=gi();
init2(q[i].s,q[i].t,i);
}
dep[]=;
tarjan(,);
for(int i=;i<=m;i++)
{
mark[q[i].s]++;
q1[q[i].lca].push_back(dep[q[i].s]);
}
dfs1(,);
for(int i=;i<=m;i++)
{
q2[q[i].t].push_back(q[i].dis-dep[q[i].t]+P);
q3[q[i].lca].push_back(q[i].dis-dep[q[i].t]+P);
}
dfs2(,);
for(int i=;i<=m;i++)if(w[q[i].lca]+dep[q[i].lca]==dep[q[i].s])ans[q[i].lca]--;
for(int i=;i<=n;i++)printf("%d ",ans[i]);
return ;
}
【NOIP2016】天天爱跑步的更多相关文章
- [NOIp2016]天天爱跑步 线段树合并
[NOIp2016]天天爱跑步 LG传送门 作为一道被毒瘤出题人们玩坏了的NOIp经典题,我们先不看毒瘤的"动态爱跑步"和"天天爱仙人掌",回归一下本来的味道. ...
- [Noip2016]天天爱跑步 LCA+DFS
[Noip2016]天天爱跑步 Description 小c同学认为跑步非常有趣,于是决定制作一款叫做<天天爱跑步>的游戏.?天天爱跑步?是一个养成类游戏,需要玩家每天按时上线,完成打卡任 ...
- 【LG1600】[NOIP2016]天天爱跑步
[LG1600][NOIP2016]天天爱跑步 题面 洛谷 题解 考虑一条路径\(S\rightarrow T\)是如何给一个观测点\(x\)造成贡献的, 一种是从\(x\)的子树内出来,另外一种是从 ...
- NOIP2016天天爱跑步 题解报告【lca+树上统计(桶)】
题目描述 小c同学认为跑步非常有趣,于是决定制作一款叫做<天天爱跑步>的游戏.«天天爱跑步»是一个养成类游戏,需要玩家每天按时上线,完成打卡任务. 这个游戏的地图可以看作一一棵包含 nn个 ...
- BZOJ4719 [Noip2016]天天爱跑步
本文版权归ljh2000和博客园共有,欢迎转载,但须保留此声明,并给出原文链接,谢谢合作. 本文作者:ljh2000作者博客:http://www.cnblogs.com/ljh2000-jump/转 ...
- noip2016天天爱跑步
题目描述 小c同学认为跑步非常有趣,于是决定制作一款叫做<天天爱跑步>的游戏.«天天爱跑步»是一个养成类游戏,需要玩家每天按时上线,完成打卡任务. 这个游戏的地图可以看作一一棵包含 个结点 ...
- bzoj 4719: [Noip2016]天天爱跑步
Description 小c同学认为跑步非常有趣,于是决定制作一款叫做<天天爱跑步>的游戏.?天天爱跑步?是一个养成类游戏,需要 玩家每天按时上线,完成打卡任务.这个游戏的地图可以看作一一 ...
- NOIP2016 天天爱跑步 80分暴力
https://www.luogu.org/problem/show?pid=1600 题目描述 小c同学认为跑步非常有趣,于是决定制作一款叫做<天天爱跑步>的游戏.«天天爱跑步»是一个养 ...
- 4719: [Noip2016]天天爱跑步
Time Limit: 40 Sec Memory Limit: 512 MB Submit: 1986 Solved: 752 [Submit][Status][Discuss] Descripti ...
- NOIP2016 天天爱跑步 线段树合并_桶_思维题
竟然独自想出来了,好开心 Code: #include<bits/stdc++.h> #define setIO(s) freopen(s".in","r&q ...
随机推荐
- Git使用方法2.0
## Git来源: 最早开始是由Ruby程序员们发起的.Ruby是日本的家伙搞出来的,日本有个代码托管网站叫heroku,当时用这个的人比较多,现在这个网站还能打开,网址是www.heroku.com ...
- 201621123062《java程序设计》第七周作业总结
1. 本周学习总结 1.1 思维导图:Java图形界面总结 1.2 可选:使用常规方法总结其他上课内容. 1.布局管理器的具体使用方法 2.事件处理模型及其代码的编写 3.Swing中的常用组件 4. ...
- 201621123040《Java程序设计》第3周学习总结
1.本周学习总结 1.1 写出你认为本周学习中比较重要的知识点关键词,如类.对象.封装等 面向对象的思想 对象 类 1.2 用思维导图或者Onenote或其他工具将这些关键词组织起来. 掌握的还不够深 ...
- 服务器数据恢复方法_存储raid硬盘离线数据恢复案例
[故障描述]某法院的一台HP-P4500的存储系统,底层是12块1TB的硬盘组的RAID.其中每6个1TB的盘一组,第一组的前面一部分组了一个RAID0+1,是存放HP-P4500嵌入式系统,接着组了 ...
- 16-TypeScript装饰器模式
在客户端脚本中,有一个类通常有一个方法需要执行一些操作,当我们需要扩展新功能,增加一些操作代码时,通常需要修改类中方法的代码,这种方式违背了开闭的原则. 装饰器模式可以动态的给类增加一些额外的职责.基 ...
- php的set_time_limit()函数
set_time_limit(0); 括号里边的数字是执行时间,如果为零说明永久执行直到程序结束,如果为大于零的数字,则不管程序是否执行完成,到了设定的秒数,程序结束. 一个简单的例子,在网页里显示1 ...
- Spring Cache扩展:注解失效时间+主动刷新缓存(二)
*:first-child { margin-top: 0 !important; } body > *:last-child { margin-bottom: 0 !important; } ...
- Web移动端适配总结
移动端适配的相关概念以及几种方案总结 适配相关概念 布局视口(layout viewport):html元素的上一级容器即顶级容器,用于解决页面在手机上显示的问题.大部分移动设备都将这个视口分辨率设置 ...
- Python内置函数(45)——ascii
英文文档: ascii(object) As repr(), return a string containing a printable representation of an object, b ...
- ejs注释问题
项目中遇到一个问题: 代码如下: 但是代码如下时,却不会出现bug: bug的导火索是ejs的注释: 因为我没有用对注释,所以被注释部分的if语句仍旧生效了. bug的根本原因是没有对mod类型进行判 ...