Description

Solution

这道题是NOIP里面最难的一道题。
暴力打的好可以拿80分,比赛的时候还是打暴力比较好。
我们思考一下从x到y的路径,这个可以拆成从x到lca的路径和从lca到y的路径,这个很明显。
如果一个点i在从x到lca 的路径可以检测到的话,那么就有deep[i]+w[i]=deep[x]。
如果一个点i在从lca到y的路径上可以检测到的话,那么就有deep[i]-w[i]=deep[y]-t(t表示x到y的路径长度)。
那么用树链剖分的方法很容易,但是很慢。有一个用桶的方法,跑得很快。
维护两个桶,一个向上的桶a和一个向下的桶b。
从x到y的一个路径,在x中a[deep[x]]加一个,当dfs把lca退栈的时候,x的影响就没有了,那么把a[deep[x]]减掉。
在lca那里需要把一个b[deep[y]]加进来,在y出栈后,就把b[deep[y]]减掉。
每次ans[x]的答案就是子树a[deep[x]+w[x]]+b[deep[x]-w[x]]的数量。
但是如果是一条链的情况,那么这样会算重,所以还要减去重复的数量。

Code

#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<cmath>
#define fo(i,a,b) for(i=a;i<=b;i++)
#define fod(i,a,b) for(i=a;i>=b;i--)
#define rep(i,a) for(i=first[a];i;i=next[i])
#define rep1(i,a) for(i=first1[a];i;i=next1[i])
#define rep2(i,a) for(i=first2[a];i;i=next2[i])
#define rep3(i,a) for(i=first3[a];i;i=next3[i])
using namespace std;
const int maxn=300007;
int i,j,k,l,t,n,m,ans[maxn],o,b;
int first[maxn*20],last[maxn*2],next[maxn*2],num;
int first1[maxn*20],last1[maxn*2],next1[maxn*2],num1;
int first2[maxn*20],last2[maxn*2],next2[maxn*2],num2;
int first3[maxn*20],last3[maxn*2],next3[maxn*2],num3;
int a[maxn],deep[maxn],f[maxn][21];
int ci[maxn];
int shang[maxn*2],xia[maxn*2];
void add(int x,int y){last[++num]=y,next[num]=first[x],first[x]=num;}
void add1(int x,int y){last1[++num1]=y,next1[num1]=first1[x],first1[x]=num1;}
void add2(int x,int y){last2[++num2]=y,next2[num2]=first2[x],first2[x]=num2;}
void add3(int x,int y){last3[++num3]=y,next3[num3]=first3[x],first3[x]=num3;}
void dfs(int x,int y){int i;deep[x]=deep[y]+1,f[x][0]=y;rep(i,x)if(last[i]!=y)dfs(last[i],x);
}
int lca(int x,int y){int i;if(deep[x]<deep[y])swap(x,y);fod(i,20,0)if(deep[f[x][i]]>deep[y])x=f[x][i];if(deep[x]!=deep[y])x=f[x][0];fod(i,20,0)if(f[x][i]!=f[y][i])x=f[x][i],y=f[y][i];return(x!=y)?f[x][0]:x;
}
void dfs1(int x,int y){int u=xia[deep[x]+a[x]],v=shang[deep[x]-a[x]+maxn],i;xia[deep[x]]+=ci[x];rep1(i,x)shang[last1[i]+maxn]++;rep(i,x)if(last[i]!=y)dfs1(last[i],x);ans[x]=xia[deep[x]+a[x]]+shang[deep[x]-a[x]+maxn]-u-v;rep2(i,x){xia[last2[i]]--;if(last2[i]==deep[x]+a[x])ans[x]--;}rep3(i,x)shang[last3[i]+maxn]--;
}
int main(){freopen("running.in","r",stdin);freopen("running.out","w",stdout);scanf("%d%d",&n,&m);fo(i,1,n-1){scanf("%d%d",&k,&l);add(k,l),add(l,k);}   dfs(1,0);fo(j,1,20)fo(i,1,n)f[i][j]=f[f[i][j-1]][j-1];fo(i,1,n)scanf("%d",&a[i]);fo(i,1,m){scanf("%d%d",&k,&l);o=lca(k,l);t=deep[k]+deep[l]-2*deep[o];ci[k]++;b=deep[l]-t;add1(l,b);add2(o,deep[k]);add3(o,b);}dfs1(1,0);fo(i,1,n)printf("%d ",ans[i]);
}

【NOIP2016提高组复赛】天天爱跑步相关推荐

  1. 【NOIP2016提高组】天天爱跑步

    题目背景 NOIP2016 提高组 Day1 T2 题目描述 小 C 同学认为跑步非常有趣,于是决定制作一款叫做<天天爱跑步>的游戏.<天天爱跑步>是一个养成类游戏,需要玩家每 ...

  2. NOIP2016提高组day2 天天爱跑步

    Description Data Constraint Solution 我们可以设一个询问[S,T]的lca为X,路径的长度为a[i],那么一个询问可以拆成[S,T]和[X,T]拆成两部分. 对于一 ...

  3. NOIP2016提高组复赛解题报告

    Day1 T1-玩具谜题 Day1 T2-天天爱跑步 Day1 T3-换教室 Day2 T1-组合数问题 Day2 T2-蚯蚓 Day2 T3-愤怒的小鸟 转载于:https://www.cnblog ...

  4. NOIP 提高组 复赛 历年 试题

    NOIP 提高组  复赛  历年 试题 NOIP 2017 提高组 复赛  试题 https://wenku.baidu.com/view/70de9e29854769eae009581b6bd97f ...

  5. P1850 [NOIP2016 提高组] 换教室

    P1850 [NOIP2016 提高组] 换教室 题意: 有2n个课安排在n个时间段上,每个时间段上都有两个一样的课同时在不同地方上,起初牛牛被所有课都被安排在Ci上课,另一节课在Di上课.牛牛现在想 ...

  6. [NOIP2016 提高组] 愤怒的小鸟

    [NOIP2016 提高组] 愤怒的小鸟 题意: 有n只猪,给出猪的坐标(xi,yi),问最少用几个形如 y=ax^2+bx 的曲线可以保证所有猪在曲线上,满足a<0,a,b为实数 n<= ...

  7. #185. [NOIP2016 提高组] 蚯蚓题解

    #185. [NOIP2016 提高组] 蚯蚓题解 题目描述 本题中,我们将用符号 ⌊c⌋\lfloor c \rfloor⌊c⌋ 表示对 ccc 向下取整,例如:⌊3.0⌋=⌊3.1⌋=⌊3.9⌋= ...

  8. NOIP2010 提高组 复赛 translate 机器翻译

    NOIP2010 提高组 复赛 translate 机器翻译 1.读题,很快弄明题意,单词不在内存中就查字典,统计查字典次数. 2.内存采用队列方式.统计进队列次数,即为查询次数. 3.程序很快编好, ...

  9. NOIP2013 提高组复赛解题报告

    NOIP2013 提高组复赛 day1 day\;1 1002. 火柴排队 贪心+数据结构/归并排序 这个"相邻交换"让我联想到了NOIP2012_day1_task2_game那 ...

最新文章

  1. 一种高效的可变行高列表行定位算法
  2. python语言及其应用下载_Python语言及其应用 中文pdf完整版[13MB]
  3. 数据中心基础设施及日常运维管理
  4. scala创建并使用Enumerations
  5. [补档]暑假集训D5总结
  6. 13000.tcp服务器简易示例程序(linux)
  7. 1-9月全国快递服务企业业务量累计完成561.4亿件 同比增长27.9%
  8. 在ubuntu 8.04下安装Oracle 11g
  9. [leetcode]求数组的第k个最大值,python快排解法
  10. 关于数论【莫比乌斯反演】
  11. 启动关闭HadoopSpark历史服务
  12. 10.机器学习sklearn-------手写数字识别实例
  13. 从0开始,html5零基础入门教程,快来看看!
  14. 在线android机型测试,免费兼容测试/MonkeyTest/100款安卓机型真机测试
  15. 互联网常见通用的运营数据指标
  16. Idea没有自动更新target目录
  17. C# dll注入(指定进程注入指定dll)
  18. android加密参数定位方法
  19. 论文解读《Co-Correcting:Noise-tolerant Medical Image Classification via mutual Label Correction》
  20. oracle itl 故障,ORACLE ITL事务槽

热门文章

  1. 剑三修复工具无法连接服务器失败怎么办,掉线3手游,你们都怎么登上的?
  2. 客服系统怎么选择?企业选择电商客服系统要从哪些方面入手?
  3. YouTuBe各类优秀频道推荐二音乐舞蹈
  4. 使用队列实现redis数据一致性的思考
  5. 善于使用F12开发人员工具来快速调试js代码
  6. Junit测试出现异常:Exception in thread “main“ java.lang.NoSuchMethodError: org.junit.runner.Description.cre
  7. Oracle数据库按每天、每周、每月、每季度、每年统计数据
  8. 三维异面直线的距离和垂足求解
  9. 魔兽世界无法连接语音聊天服务器,玩家感悟:你不知道的那个“上不了语音的人” - 178魔兽世界专区...
  10. pagehelper联表分页查询