Description

小c同学认为跑步非常有趣,于是决定制作一款叫做《天天爱跑步》的游戏。?天天爱跑步?是一个养成类游戏,需要
玩家每天按时上线,完成打卡任务。这个游戏的地图可以看作一一棵包含 N个结点和N-1 条边的树, 每条边连接两
个结点,且任意两个结点存在一条路径互相可达。树上结点编号为从1到N的连续正整数。现在有个玩家,第个玩家的
起点为Si ,终点为Ti  。每天打卡任务开始时,所有玩家在第0秒同时从自己的起点出发, 以每秒跑一条边的速度,
不间断地沿着最短路径向着自己的终点跑去, 跑到终点后该玩家就算完成了打卡任务。 (由于地图是一棵树, 所以
每个人的路径是唯一的)小C想知道游戏的活跃度, 所以在每个结点上都放置了一个观察员。 在结点的观察员会选
择在第Wj秒观察玩家, 一个玩家能被这个观察员观察到当且仅当该玩家在第Wj秒也理到达了结点J  。 小C想知道
每个观察员会观察到多少人?注意: 我们认为一个玩家到达自己的终点后该玩家就会结束游戏, 他不能等待一 段时
间后再被观察员观察到。 即对于把结点J作为终点的玩家: 若他在第Wj秒重到达终点,则在结点J的观察员不能观察
到该玩家;若他正好在第Wj秒到达终点,则在结点的观察员可以观察到这个玩家。

Input

第一行有两个整数N和M 。其中N代表树的结点数量, 同时也是观察员的数量, M代表玩家的数量。
接下来n-1 行每行两个整数U和V ,表示结点U 到结点V 有一条边。
接下来一行N 个整数,其中第个整数为Wj , 表示结点出现观察员的时间。
接下来 M行,每行两个整数Si和Ti,表示一个玩家的起点和终点。
对于所有的数据,保证 。
1<=Si,Ti<=N,0<=Wj<=N

Output

输出1行N 个整数,第个整数表示结点的观察员可以观察到多少人。

Sample Input

6 3
2 3
1 2
1 4
4 5
4 6
0 2 5 1 2 3
1 5
1 3
2 6

Sample Output

1 2 1 0 1

HINT

对于1号点,Wi=0,故只有起点为1号点的玩家才会被观察到,所以玩家1和玩家2被观察到,共有2人被观察到。
对于2号点,没有玩家在第2秒时在此结点,共0人被观察到。
对于3号点,没有玩家在第5秒时在此结点,共0人被观察到。
对于4号点,玩家1被观察到,共1人被观察到。
对于5号点,玩家1被观察到,共1人被观察到。
对于6号点,玩家3被观察到,共1人被观察到。

Source

昨天终于把这个史前巨坑给填了

对于一个点能否产生贡献我们分两种(拆路径)情况讨论:

1.在s--lca间,那么T[i]=deep[s]-deep[i],移项:T[i]+deep[i]=deep[s];

1.在lca---t之间,那么T[i]=deep[s]+deep[i]-2*deep[lca],移项:deep[i]-T[i]=2*deep[lca]-deep[s];

我们发现等式右边只和i有关,我们相当于要把路径上某些满足条件的点加1

于是好像用路径分块+桶可以过掉95分,(n*sqrt(n)*logn);

正解是用了一种类似于树上差分打标记的方法(在s和t加,lca处减掉),具体思想比较巧妙,用差分后相当于查询子树和(查询桶中特定值),具体实现不太好讲,代码比较清楚。

分块:

  1. // MADE BY QT666
  2. #include<cstdio>
  3. #include<algorithm>
  4. #include<cmath>
  5. #include<iostream>
  6. #include<cstring>
  7. #define RG register
  8. using namespace std;
  9. typedef long long ll;
  10. const int N=100050;
  11. int gi()
  12. {
  13. int x=0,flag=1;
  14. char ch=getchar();
  15. while(ch<'0'||ch>'9'){if(ch=='-') flag=-1;ch=getchar();}
  16. while(ch>='0'&&ch<='9') x=x*10+ch-'0',ch=getchar();
  17. return x*flag;
  18. }
  19. int head[N*2],to[N*2],nxt[N*2];
  20. int deep[N],size[N],son[N],top[N],fa[N],cnt,T[N],dfn[N],pd[2][N];
  21. int n,m,sum,pos[N],block,L[N],R[N],tt,id[N];
  22. int tong[320][N*2][2],ans[N],tot;
  23. struct data{
  24. int l,r,flag;
  25. }p[N];
  26. inline void lnk(int x,int y){
  27. to[++cnt]=y,nxt[cnt]=head[x],head[x]=cnt;
  28. to[++cnt]=x,nxt[cnt]=head[y],head[y]=cnt;
  29. }
  30. inline void dfs1(RG int x,RG int f){
  31. deep[x]=deep[f]+1;size[x]=1;
  32. for(RG int i=head[x];i;i=nxt[i]){
  33. int y=to[i];
  34. if(y!=f){
  35. fa[y]=x;dfs1(y,x);
  36. size[x]+=size[y];
  37. if(size[y]>size[son[x]]) son[x]=y;
  38. }
  39. }
  40. }
  41. inline void dfs2(RG int x,RG int f){
  42. dfn[x]=++sum;id[sum]=x;top[x]=f;
  43. if(son[x]) dfs2(son[x],f);
  44. for(RG int i=head[x];i;i=nxt[i]){
  45. int y=to[i];
  46. if(y!=fa[x]&&y!=son[x]) dfs2(y,y);
  47. }
  48. }
  49. inline int LCA(RG int x,RG int y){
  50. tot=0;int fl=1;
  51. while(top[x]!=top[y]){
  52. if(deep[top[x]]<deep[top[y]]) swap(x,y),fl^=1;
  53. p[++tot]=(data){dfn[top[x]],dfn[x],fl};
  54. x=fa[top[x]];
  55. }
  56. if(deep[x]<deep[y]) swap(x,y),fl^=1;
  57. p[++tot]=(data){dfn[y],dfn[x],fl};
  58. return y;
  59. }
  60. inline void update(RG int l,RG int r,RG int x,int flag){
  61. if(pos[l]==pos[r]){
  62. for(RG int i=l;i<=r;i++){
  63. if(pd[flag][id[i]]==x) ans[id[i]]++;
  64. }
  65. }
  66. else{
  67. for(RG int i=pos[l]+1;i<pos[r];i++) tong[i][x][flag]++;
  68. for(RG int i=l;i<=R[pos[l]];i++){
  69. if(pd[flag][id[i]]==x) ans[id[i]]++;
  70. }
  71. for(RG int i=L[pos[r]];i<=r;i++){
  72. if(pd[flag][id[i]]==x) ans[id[i]]++;
  73. }
  74. }
  75. }
  76. int main(){
  77. freopen("running.in","r",stdin);
  78. freopen("running.out","w",stdout);
  79. n=gi(),m=gi();
  80. for(RG int i=1;i<n;i++){
  81. int x=gi(),y=gi();lnk(x,y);
  82. }
  83. dfs1(1,0),dfs2(1,1);
  84. block=sqrt(n);tt=n/block;
  85. if(n%block) tt++;
  86. for(RG int i=1;i<=n;i++) pos[i]=(i-1)/block+1;
  87. for(RG int i=1;i<=tt;i++) L[i]=(i-1)*block+1,R[i]=i*block;
  88. R[tt]=n;
  89. for(RG int i=1;i<=n;i++) T[i]=gi(),pd[0][i]=deep[i]-T[i]+N,pd[1][i]=deep[i]+T[i]+N;
  90. for(RG int i=1;i<=m;i++){
  91. int s=gi(),t=gi(),lca=LCA(s,t);
  92. for(RG int j=1;j<=tot;j++){
  93. if(p[j].flag==1) update(p[j].l,p[j].r,deep[s]+N,1);
  94. else update(p[j].l,p[j].r,2*deep[lca]-deep[s]+N,0);
  95. }
  96. }
  97. for(RG int i=1;i<=n;i++){
  98. printf("%d ",ans[i]+tong[pos[dfn[i]]][pd[0][i]][0]+tong[pos[dfn[i]]][pd[1][i]][1]);
  99. }
  100. }

正解:

  1. // MADE BY QT666
  2. #include<cstdio>
  3. #include<algorithm>
  4. #include<cmath>
  5. #include<iostream>
  6. #include<vector>
  7. #include<cstring>
  8. #define RG register
  9. using namespace std;
  10. typedef long long ll;
  11. const int N=500050;
  12. int gi()
  13. {
  14. int x=0,flag=1;
  15. char ch=getchar();
  16. while(ch<'0'||ch>'9'){if(ch=='-') flag=-1;ch=getchar();}
  17. while(ch>='0'&&ch<='9') x=x*10+ch-'0',ch=getchar();
  18. return x*flag;
  19. }
  20. int head[N*2],to[N*2],nxt[N*2];
  21. int deep[N],size[N],son[N],top[N],fa[N],cnt,T[N],dfn[N];
  22. int n,m,sum,tt,id[N],ans[N],maxn;
  23. int tong[2*N],tong2[2*N],num[N];
  24. vector<int>a[N],b[N],c[N];
  25. struct data{
  26. int s,t,lca,len;
  27. }q[N];
  28. inline void lnk(int x,int y){
  29. to[++cnt]=y,nxt[cnt]=head[x],head[x]=cnt;
  30. to[++cnt]=x,nxt[cnt]=head[y],head[y]=cnt;
  31. }
  32. inline void dfs1(RG int x,RG int f){
  33. deep[x]=deep[f]+1;size[x]=1;
  34. maxn=max(deep[x],maxn);
  35. for(RG int i=head[x];i;i=nxt[i]){
  36. int y=to[i];
  37. if(y!=f){
  38. fa[y]=x;dfs1(y,x);
  39. size[x]+=size[y];
  40. if(size[y]>size[son[x]]) son[x]=y;
  41. }
  42. }
  43. }
  44. inline void dfs2(RG int x,RG int f){
  45. dfn[x]=++sum;id[sum]=x;top[x]=f;
  46. if(son[x]) dfs2(son[x],f);
  47. for(RG int i=head[x];i;i=nxt[i]){
  48. int y=to[i];
  49. if(y!=fa[x]&&y!=son[x]) dfs2(y,y);
  50. }
  51. }
  52. inline int LCA(RG int x,RG int y){
  53. while(top[x]!=top[y]){
  54. if(deep[top[x]]<deep[top[y]]) swap(x,y);
  55. x=fa[top[x]];
  56. }
  57. if(deep[x]<deep[y]) swap(x,y);
  58. return y;
  59. }
  60. inline void DFS_up(int x,int f){
  61. int now=T[x]+deep[x],last=tong[now+N];
  62. for(int i=head[x];i;i=nxt[i]) {
  63. int y=to[i];
  64. if(y!=f) DFS_up(y,x);
  65. }
  66. tong[deep[x]+N]+=num[x];
  67. ans[x]=tong[now+N]-last;
  68. for(int i=0;i<a[x].size();i++) tong[deep[a[x][i]]+N]--;
  69. }
  70.  
  71. inline void DFS_down(int x,int f){
  72. int now=deep[x]-T[x],last=tong2[now+N];
  73. for(int i=head[x];i;i=nxt[i]) {
  74. int y=to[i];
  75. if(y!=f) DFS_down(y,x);
  76. }
  77. for(int i=0;i<b[x].size();i++) tong2[N+b[x][i]]++;
  78. ans[x]+=tong2[now+N]-last;
  79. for(int i=0;i<c[x].size();i++) tong2[N+c[x][i]]--;
  80. }
  81. int main(){
  82. freopen("running.in","r",stdin);
  83. freopen("running.out","w",stdout);
  84. n=gi(),m=gi();
  85. for(RG int i=1;i<n;i++){
  86. int x=gi(),y=gi();lnk(x,y);
  87. }
  88. dfs1(1,0),dfs2(1,1);
  89. for(int i=1;i<=n;i++) T[i]=gi();
  90. for(int i=1;i<=m;i++){
  91. q[i].s=gi(),q[i].t=gi();num[q[i].s]++;
  92. q[i].lca=LCA(q[i].s,q[i].t);
  93. q[i].len=deep[q[i].s]+deep[q[i].t]-2*deep[q[i].lca];
  94. a[q[i].lca].push_back(q[i].s);
  95. }
  96. DFS_up(1,0);
  97. for(int i=1;i<=m;i++){
  98. b[q[i].t].push_back(deep[q[i].t]-q[i].len);
  99. c[q[i].lca].push_back(deep[q[i].t]-q[i].len);
  100. }
  101. DFS_down(1,0);
  102. for(int i=1;i<=m;i++) if(deep[q[i].s]-deep[q[i].lca]==T[q[i].lca]) ans[q[i].lca]--;
  103. printf("%d",ans[1]);
  104. for(int i=2;i<=n;i++) printf(" %d",ans[i]);
  105. }

bzoj 4719: [Noip2016]天天爱跑步的更多相关文章

  1. BZOJ 4719 [Noip2016]天天爱跑步 ——树链剖分

    一直以为自己当时是TLE了,但是再看发现居然WA? 然后把数组扩大一倍,就A掉了.QaQ 没什么好说的.一段路径分成两段考虑,上升的一段深度+时间是定值,下降的一段深度-时间是定值,然后打标记统计即可 ...

  2. bzoj 4719: [Noip2016]天天爱跑步【树上差分+dfs】

    长久以来的心理阴影?但是其实非常简单-- 预处理出deep和每组st的lca,在这里我简单粗暴的拿树剖爆算了 然后考虑对于一组s t lca来说,被这组贡献的观察员x当且仅当: x在s到lca的路径上 ...

  3. 4719: [Noip2016]天天爱跑步

    Time Limit: 40 Sec Memory Limit: 512 MB Submit: 1986 Solved: 752 [Submit][Status][Discuss] Descripti ...

  4. [NOIp2016]天天爱跑步 线段树合并

    [NOIp2016]天天爱跑步 LG传送门 作为一道被毒瘤出题人们玩坏了的NOIp经典题,我们先不看毒瘤的"动态爱跑步"和"天天爱仙人掌",回归一下本来的味道. ...

  5. [Noip2016]天天爱跑步 LCA+DFS

    [Noip2016]天天爱跑步 Description 小c同学认为跑步非常有趣,于是决定制作一款叫做<天天爱跑步>的游戏.?天天爱跑步?是一个养成类游戏,需要玩家每天按时上线,完成打卡任 ...

  6. 【LG1600】[NOIP2016]天天爱跑步

    [LG1600][NOIP2016]天天爱跑步 题面 洛谷 题解 考虑一条路径\(S\rightarrow T\)是如何给一个观测点\(x\)造成贡献的, 一种是从\(x\)的子树内出来,另外一种是从 ...

  7. NOIP2016天天爱跑步 题解报告【lca+树上统计(桶)】

    题目描述 小c同学认为跑步非常有趣,于是决定制作一款叫做<天天爱跑步>的游戏.«天天爱跑步»是一个养成类游戏,需要玩家每天按时上线,完成打卡任务. 这个游戏的地图可以看作一一棵包含 nn个 ...

  8. BZOJ4719 [Noip2016]天天爱跑步

    本文版权归ljh2000和博客园共有,欢迎转载,但须保留此声明,并给出原文链接,谢谢合作. 本文作者:ljh2000作者博客:http://www.cnblogs.com/ljh2000-jump/转 ...

  9. noip2016天天爱跑步

    题目描述 小c同学认为跑步非常有趣,于是决定制作一款叫做<天天爱跑步>的游戏.«天天爱跑步»是一个养成类游戏,需要玩家每天按时上线,完成打卡任务. 这个游戏的地图可以看作一一棵包含 个结点 ...

随机推荐

  1. ES7前端异步玩法:async/await理解

    在最新的ES7(ES2017)中提出的前端异步特性:async.await. 什么是async.await? async顾名思义是"异步"的意思,async用于声明一个函数是异步的 ...

  2. Asp.net Core中SignalR Core预览版的一些新特性前瞻,附源码(消息订阅与发送二进制数据)

    目录 SignalR系列目录(注意,是ASP.NET的目录.不是Core的) 前言 一晃一个月又过去了,上个月有个比较大的项目要验收上线.所以忙的脚不沾地.现在终于可以忙里偷闲,写一篇关于Signal ...

  3. 批量抓取cisco设备配置脚本编写(expect/sed/awk/shell)

    应同事需求自行编写了第一个脚本,中间遇到一些坑. 需求,要求抓取设备"show ip interface brief"信息和"show interface des&quo ...

  4. 基于‘BOSS直聘的招聘信息’分析企业到底需要什么样的PHP程序员

    原文地址:http://www.jtahstu.com/blog/scrapy_zhipin_php.html 基于'BOSS直聘的招聘信息'分析企业到底需要什么样的PHP程序员 标签(空格分隔): ...

  5. Express4.x API (二):Request (译)

    写在前面 最近学习express想要系统的过一遍API,www.expressjs.com是express英文官网(进入www.epxressjs.com.cn发现也是只有前几句话是中文呀~~),所以 ...

  6. Oracle442个应用场景-----------角色管理

    --------------------------------角色管理------------------------------------ 一.角色的概念和特性 1.什么是角色? 角色就是相关权 ...

  7. iOS 力学动画生成器UIKit Dynamics 之碰撞效果解说

    UIKit Dynamic是iOS7 新增的一组类和方法.可赋予UIView逼真的行为和特征,不须要写动画效果那些繁琐的代码,让开发者可以轻松地改善应用的用户体验.一共同拥有6个可用于定制UIDyna ...

  8. NanUI 0.4.4发布

    NanUI是一个基于ChromiumFX开源项目的.Net Winform界面库,ChromiumFX是Chromium Embedded Framework的.Net实现.众所周知,Chromium ...

  9. Intellij 如何在新窗口中打开项目

    好多程序员都使用intelliJ idea开发项目,由于不小心设置了不在提示打开项目项目方式,所以只能打开一个窗口,而且只能同时打开一个项目,特别郁闷,分享下设置后的效果 工具/原料   Intell ...

  10. iOS设计模式之懒加载

    一.为什么要懒加载? 答: iPhone设备内存有限,如果在程序在启动后就一次性加载将来会用到的所有资源,那么久可能会耗尽iOS设备的内存.这些资源例如大量的数据,图片,音频,过多的控件等. 二.懒加 ...