题面

​ 直接写正解吧, 不想再写部分分了, 对于\(u\)和\(v\), 我们可以将它拆成两条路径, \(u\)到\(lca(u, v)\)和\(lca(u, v)\)到v, 在这里只分析从\(u\)到\(lca(u, v)\)的路径(其实是我不想写).

​ 对于一个点\(u\), 设它的深度为\(dep[u]\), 值为\(dep[i] + w[i]\)的物品有\(cnt[w[i] + dep[i]]\)个, 这样的话, 只有在满足\(dep[u] - dep[i] = w[i]\)的时候, 才能对这个点产生贡献, 由于这三个数都是确定的, 所以在某个起点出发到一个终点结束的路径, 我们处理出他的lca后, 可以使用树上差分, 将一个值为\(dep[u]\)的物品从\(u\)传入, 从\(lca(u, v)\)的父亲传出, 这样就可以对每个点进行树上差分, 在插入之前记录一下\(cnt[w[i] + dep[i]]\)的物品有多少个, 再从底往上更新, 查询当前点\(dep[i] + w[i]\)的物品有多少个, 相减即可.

​ 至于\(lca(u, v)\)的情况, 大家根据上面列一个式子用同样的方法判断就行了, 这里只列出式子, 大家也可以自己推导一下, 最后是这个样子\(dep[u] + dep[i] - 2 * dep[lca(u, v)] = w[i]\), 有可能等式左边的会小于零, 在数组上平移一段即可.

具体代码

#include <iostream>
#include <cstdio>
#include <vector>
#define N 300005
using namespace std; int n, m, w[N], head[N], cnt, f[N][20], dep[N], num[N], c1[N], c2[N << 1];
struct node
{
int to, next;
} edge[N << 1];
vector<int> lin[N], lout[N], rin[N], rout[N]; inline int read()
{
int x = 0, w = 1;
char c = getchar();
while(c < '0' || c > '9') { if (c == '-') w = -1; c = getchar(); }
while(c >= '0' && c <= '9') { x = x * 10 + c - '0'; c = getchar(); }
return x * w;
} inline void add(int u, int v) { edge[++cnt].to = v; edge[cnt].next = head[u]; head[u] = cnt; } void dfs(int u, int fa)
{
dep[u] = dep[fa] + 1;
for(int i = 1; i <= 18 && f[f[u][i - 1]][i - 1]; i++)
f[u][i] = f[f[u][i - 1]][i - 1];
for(int i = head[u]; i; i = edge[i].next)
{
int v = edge[i].to; if(v == fa) continue;
f[v][0] = u; dfs(v, u);
}
} int LCA(int u, int v)
{
if(dep[u] > dep[v]) swap(u, v);
for(int i = 18; i >= 0; i--)
if(dep[f[v][i]] >= dep[u]) v = f[v][i];
if(v == u) return u;
for(int i = 18; i >= 0; i--)
if(f[u][i] != f[v][i]) { u = f[u][i]; v = f[v][i]; }
return f[u][0];
} void down(int u)
{
int sum1 = c1[w[u] + dep[u]], sum2 = c2[w[u] - dep[u] + 300000];
for(int i = head[u]; i; i = edge[i].next)
if(edge[i].to != f[u][0]) down(edge[i].to);
for(unsigned int i = 0; i < lin[u].size(); i++) c1[lin[u][i]]++;//记录一下
for(unsigned int i = 0; i < lout[u].size(); i++) c1[lout[u][i]]--;
for(unsigned int i = 0; i < rin[u].size(); i++) c2[rin[u][i] + 300000]++;
for(unsigned int i = 0; i < rout[u].size(); i++) c2[rout[u][i] + 300000]--;//记得平移
num[u] += c1[w[u] + dep[u]] - sum1 + c2[w[u] - dep[u] + 300000] - sum2; //相减即为答案
} int main()
{
n = read(); m = read();
for(int i = 1; i < n; i++)
{
int u = read(), v = read();
add(u, v); add(v, u);
}
for(int i = 1; i <= n; i++) w[i] = read();
dfs(1, 0);
for(int i = 1; i <= m; i++)
{
int u = read(), v = read();
int lca = LCA(u, v);
lin[u].push_back(dep[u]); lout[f[lca][0]].push_back(dep[u]); rin[v].push_back(dep[u] - 2 * dep[lca]); rout[lca].push_back(dep[u] - 2 * dep[lca]); //u到lca和lca到v, 由于lca已经走过, 这个时候从lca传入就相当于这个物品是在lca与v那条链上lca的儿子传入, 差分嘛
}
down(1);
for(int i = 1; i <= n; i++) printf("%d%c", num[i], i == n ? '\n' : ' ');
return 0;
}

[luogu1600] 天天爱跑步的更多相关文章

  1. [luogu1600 noip2016] 天天爱跑步 (树上差分)

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

  2. UOJ261 【NOIP2016】天天爱跑步

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

  3. BZOJ4719 [Noip2016]天天爱跑步

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

  4. noip2016天天爱跑步

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

  5. bzoj 4719: [Noip2016]天天爱跑步

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

  6. [NOIP]2016天天爱跑步

    [NOIP]2016天天爱跑步 标签: LCA 树上差分 NOIP Description 小C同学认为跑步非常有趣,于是决定制作一款叫做<天天爱跑步>的游戏.<天天爱跑步>是 ...

  7. NOIP2016 天天爱跑步 80分暴力

    https://www.luogu.org/problem/show?pid=1600 题目描述 小c同学认为跑步非常有趣,于是决定制作一款叫做<天天爱跑步>的游戏.«天天爱跑步»是一个养 ...

  8. [NOIp 2016]天天爱跑步

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

  9. 【NOIP2016】【LCA】【树上差分】【史诗级难度】天天爱跑步

    学弟不是说要出丧题吗>>所以我就研究了1天lca又研究了1天tj然后研究了一天天天爱跑步,终于写了出来.(最后的平均用时为240ms...比学弟快了1倍...) 题意:给你颗树,然后有m个 ...

随机推荐

  1. UVAlive6807 Túnel de Rata (最小生成树)

    题意 题目链接 Sol 神仙题Orz 我们考虑选的边的补集,可以很惊奇的发现,这个补集中的边恰好是原图中的一颗生成树: 并且答案就是所有边权的和减去这个边集中的边的权值: 于是我们只需要求最大生成树就 ...

  2. Dynamics 365Online 查询Web Api的请求WebUri

    在on-premises版本中,获取weburi的方式是进设置-自定义项-开发人员资源中查看地址,但online版本中的地址会有些许的差异 online的开发者资源中的地址如下图,如果你在页面java ...

  3. CentOS7.4 + Hadoop2.9安装配置管理(分布式)

    1.  规划 1.1.  机器列表 NameNode SecondaryNameNode DataNodes 192.168.1.121 192.168.1.122 192.168.1.101 192 ...

  4. 报表在vista和win7下无法浏览应用的解决办法

     对于vista和win7系统,报表工具有着良好的兼容性,无论是设计器还是实际应用.有些客户在安装报表设计报表的时候没有遇到问题,但是在这两种系统下会发现无法启动应用,或者打开设计器自带的ie浏览 ...

  5. CSS 样式初始化

    去除浏览器对html的附加样式,避免不同浏览器之间的样式差异,给前端开发提供统一的样式基础.附加样式: .clearfix - 清除浮动 .wordsBreak - 允许文本在任意位置的换行 .ell ...

  6. java 内存分析之堆栈空间

    package Demo; public class Demo { public static void main(String[] args) { Demo demo = new Demo(); ; ...

  7. 0java之泛型解说

    1.集合中只能装入引用数据类型,不能装入基本数据类型.如,装入int类型的数值123会自动装箱. 2.开发人员装入集合的数据类型不确定,所以它被设计成可以装入所有的Object. 3.新的问题产生,装 ...

  8. oracle常见的等待事件说明

    转自 http://blog.itpub.net/29371470/viewspace-1063994/ 1. Buffer busy waits 从本质上讲,这个等待事件的产生仅说明了一个会话在等待 ...

  9. Linux 系统下 centOS 7 ipconfig 提示没有安装

    首先更正一下,在Linux系统下,查看IP地址,指令是ifconfig 没有root权限情况下,安装指令为 sudo yum -y install net-tool 有root权限的话,直接执行 yu ...

  10. redis介绍(6)集群(ruby)

    redis集群: redis集群是高可用的一种体现,让整个redis圈更加稳定,不易出现宕机的情况, redis原理: redis3.0之前是不支持集群的,实现集群要自己去配置实现,很麻烦,在3.0之 ...