天天放毒...

首先介绍一个树上差分。

每次进入的时候记录贡献,跟出来的时候的差值就是子树贡献。

然后就可以做了。

发现考虑每个人的贡献有困难。

于是考虑每个观察员的答案。

把路径拆成两条,以lca分开。x -> z -> y,完全分成A,B两部分。

那么A:d[x] = w[z] + d[z];B:len - d[y] + N = w[z] - d[z] + N;

这里+ N是为了防止负数。

然后发现右边只跟z有关,这里的z可以是路径上任一点。

那么对于每个人,把需要树上差分统计的左边数值用vector记录。

然后跑一遍统计即可。

注意:如果lca能观测到这条路径,那么--,因为测了A,B两次。

看代码。

 #include <cstdio>
#include <vector> const int N = ; inline void read(int &x) {
x = ;
char c = getchar();
while(c < '' || c > '') {
c = getchar();
}
while(c >= '' && c <= '') {
x = (x << ) + (x << ) + c - ;
c = getchar();
}
return;
} struct Edge {
int v, nex;
}edge[N << ]; int top; int n, m, e[N], w[N], binA[N << ], binB[N << ], cancel[N], d[N], fa[N][], lm, ans[N];
std::vector<int> vA[N], vB[N], vA_[N], vB_[N]; inline void add(int x, int y) {
edge[++top].v = y;
edge[top].nex = e[x];
e[x] = top;
return;
} void DFS1(int x, int f) {
fa[x][] = f;
d[x] = d[f] + ;
for(int i = e[x]; i; i = edge[i].nex) {
int y = edge[i].v;
if(y != f) {
DFS1(y, x);
}
}
return;
} inline void getlca() {
DFS1(, );
while(( << lm) < n) {
lm++;
}
for(int i = ; i <= lm; i++) {
for(int x = ; x <= n; x++) {
fa[x][i] = fa[fa[x][i - ]][i - ];
}
}
return;
} inline int lca(int x, int y) {
int t = lm;
while(d[x] > d[y]) {
std::swap(x, y);
}
while(t > - && d[y] > d[x]) {
if(d[fa[y][t]] >= d[x]) {
y = fa[y][t];
}
t--;
}
if(x == y) {
return x;
}
t = lm;
while(t > - && fa[x][] != fa[y][]) {
if(fa[x][t] != fa[y][t]) {
x = fa[x][t];
y = fa[y][t];
}
t--;
}
return fa[x][];
} void DFS(int x) {
int A = binA[w[x] + d[x]];
int B = binB[w[x] - d[x] + N];
for(int i = e[x]; i; i = edge[i].nex) {
int y = edge[i].v;
if(y != fa[x][]) {
DFS(y);
}
}
for(int i = ; i < vA[x].size(); i++) {
binA[vA[x][i]]++;
}
for(int i = ; i < vB[x].size(); i++) {
binB[vB[x][i]]++;
}
ans[x] += binA[w[x] + d[x]] - A;
ans[x] += binB[w[x] - d[x] + N] - B;
for(int i = ; i < vA_[x].size(); i++) {
binA[vA_[x][i]]--;
}
for(int i = ; i < vB_[x].size(); i++) {
binB[vB_[x][i]]--;
}
return;
} int main() {
read(n);
read(m);
int x, y, z;
for(int i = ; i < n; i++) {
read(x);
read(y);
add(x, y);
add(y, x);
}
for(int i = ; i <= n; i++) {
read(w[i]);
}
getlca();
for(int i = ; i <= m; i++) {
read(x);
read(y);
z = lca(x, y);
vA[x].push_back(d[x]); //d[x] = w[z] + d[z];
vA_[z].push_back(d[x]); //
int len = d[x] + d[y] - * d[z];
vB[y].push_back(len - d[y] + N); //len - d[y] + N = w[z] - d[z] + N;
vB_[z].push_back(len - d[y] + N);
if(d[x] == w[z] + d[z]) {
cancel[z]++;
}
} DFS(); for(int i = ; i <= n; i++) {
printf("%d ", ans[i] - cancel[i]);
}
/*puts("");
for(int i = 1; i < N * 2; i++) {
if(binB[i]) {
printf("binB[%d] = %d\n", i, binB[i]);
}
if(binA[i]) {
printf("binA[%d] = %d\n", i, binA[i]);
}
}
printf("over");*/ return ;
}

AC代码

zbtrs大佬用主席树A了,还说显然是主席树,太强了%%%

洛谷P1600 天天爱跑步的更多相关文章

  1. 洛谷P1600 天天爱跑步(线段树合并)

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

  2. 洛谷 P1600 天天爱跑步

    https://www.luogu.org/problemnew/show/P1600 (仅做记录) 自己的假方法: 每一次跑从a到b:设l=lca(a,b)对于以下产生贡献: a到l的链上所有的点( ...

  3. 洛谷P1600 天天爱跑步——树上差分

    题目:https://www.luogu.org/problemnew/show/P1600 看博客:https://blog.csdn.net/clove_unique/article/detail ...

  4. 洛谷P1600 天天爱跑步(差分 LCA 桶)

    题意 题目链接 Sol 一步一步的来考虑 \(25 \%\):直接\(O(nm)\)的暴力 链的情况:维护两个差分数组,分别表示从左向右和从右向左的贡献, \(S_i = 1\):统计每个点的子树内有 ...

  5. 洛谷 P1600 天天爱跑步(LCA+乱搞)

    传送门 我们把每一条路径拆成$u->lca$和$lca->v$的路径 先考虑$u->lca$,如果这条路径会对路径上的某一个点产生贡献,那么满足$dep[u]-dep[x]=w[x] ...

  6. 洛谷P1600 天天爱跑步——题解

    题目传送 首先要考虑入手点.先考虑一个一个玩家处理,显然不加优化的话,时间复杂度是O(n)的.发现对于玩家路径上的点都有一个观察员,一个都不能忽视,看起来是很难优化了.在做题时,发现一个思路很难想,就 ...

  7. 洛谷$P1600$ 天天爱跑步 树上差分

    正解:树上差分 解题报告: 传送门$QwQ$! 这题还挺妙的,,,我想了半天才会$kk$ 首先对一条链$S-T$,考虑先将它拆成$S-LCA$和$LCA-T$,分别做.因为总体上来说差不多接下来我就只 ...

  8. [NOIP 2016D2T2/Luogu P1600] 天天爱跑步 (LCA+差分)

    待填坑 Code //Luogu P1600 天天爱跑步 //Apr,4th,2018 //树上差分+LCA #include<iostream> #include<cstdio&g ...

  9. AC日记——天天爱跑步 洛谷 P1600

    天天爱跑步 思路: 树上差分+分层动态线段树: (伏地膜,跪烂xxy) 代码: #include <bits/stdc++.h> using namespace std; #define ...

随机推荐

  1. better-scroll的参数和方法

    格式:let obj = new BScroll(object,{[option1,],.,.}); 注意,如果在某一个组件内创建了一个BScroll的实例,在组件生命周期结束前要注意调用destro ...

  2. 2-Twenty third Scrum Meeting-20151229

    前言 因为服务器关闭至今,我们的开发项目也遭遇停滞一个星期.与网站开发负责人员协商之后,今天继续开放服务器.我们的项目也能够继续下去.比规定的开发时间(截止为2015/12/29)推迟,因此我们今天又 ...

  3. struts引入s标签

    <%@ taglib prefix="s" uri="/struts-tags"%>

  4. Docker 部署Gitlab

    sudo docker run -d \ -h 192.168.16.88 \ -p 89:80 -p 23:22 \ --name gitlab \ --restart always \ --vol ...

  5. Linux 重启网络提示找不到eth0(no device found for “System eth0”)

    一.背景 使用VMWare创建了一个虚拟机(VM1),然后通过拷贝的方式创建了另一台虚拟机(VM2).在第二台虚拟机上设置网卡为固定IP,使用service network restart重启网络的时 ...

  6. HTML5 & auto download image

    HTML5 & auto download image https://www.sitepoint.com/new-html5-attributes-hyperlinks-download-m ...

  7. Python中print/format字符串格式化实例

    Python 字符串格式化使用 "字符 %格式1 %格式2 字符"%(变量1,变量2),%格式表示接受变量的类型.简单的使用例子如下 # 例:字符串格式化Name = '17jo' ...

  8. sql 表,字段(列),表数据(行)相关命令

    随便转载,保留出处:http://www.cnblogs.com/aaron-agu/ 注: 以下所有操作都在登陆数据库后执行 命令use test;  test为数据库名 查看表 show tabl ...

  9. Database testing test scenarios

    1 check if correct data is getting saved is database upon successful page submit2 check values for c ...

  10. 我终于激活Windows Server2008 R2了!!

    经过我不懈的努力,在重装两次系统后,我终于实现了win2008的KMS激活.这个方法可以避免虚拟机架设KMS服务器的麻烦.现将激活方法发布如下. 首先要选择安装的操作系统.Windows Server ...