LG1600 天天爱跑步
题意

分析
对一个(s,t)查询,令f=lca(s,t),则操作可化为(s,f),(f,t)。
考虑观察到的情况,若x在s到t的路径上,且x观察到,则
\[
\textrm{dep}_s-\textrm{dep}_x=w_x\\
\textrm{dep}_s=\textrm{dep}_x+w_x
\]
或者
\[
\textrm{dep}_t+\textrm{dep}_s-2\textrm{dep}_f-(\textrm{dep}_t-\textrm{dep}_x)=w_x\\
\textrm{dep}_s-2\textrm{dep}_f=w_x-\textrm{dep}_x
\]
那么相当于在x的子树中查询等于x相关的值的个数。
考虑树上差分,在s或t处加1,在f处减1,然后对每个权值建一个以dfn序为下标的线段树,维护加减1的个数。
枚举树上节点,查询子树即可。
时间复杂度\(O(m \log n + n \log n)\)
代码
注意数组的范围。
分成2条链操作的时候,1次统计f,1次不统计f。
第2种情况时为了数组下标大于0,要加上n+1。
注意fa[f]=0的情况,不应该操作线段树。
#include<cstdlib>
#include<cstdio>
#include<cmath>
#include<cstring>
#include<ctime>
#include<iostream>
#include<string>
#include<vector>
#include<list>
#include<deque>
#include<stack>
#include<queue>
#include<map>
#include<set>
#include<bitset>
#include<algorithm>
#include<complex>
#define rg register
#define il inline
#pragma GCC optimize ("O3")
using namespace std;
template<class T> inline T read(T&x)
{
T data=0;
int w=1;
char ch=getchar();
while(!isdigit(ch))
{
if(ch=='-')
w=-1;
ch=getchar();
}
while(isdigit(ch))
data=10*data+ch-'0',ch=getchar();
return x=data*w;
}
typedef long long ll;
const int INF=0x7fffffff;
const int MAXN=3e5+7;
int w[MAXN],t[MAXN];
struct Edge
{
int nx,to;
}E[MAXN<<1];
int head[MAXN],ecnt;
il void addedge(rg int x,rg int y)
{
E[++ecnt].to=y;
E[ecnt].nx=head[x],head[x]=ecnt;
}
int fa[MAXN],dep[MAXN],siz[MAXN],son[MAXN];
il void dfs1(rg int x,rg int f)
{
fa[x]=f,dep[x]=dep[f]+1,siz[x]=1;
for(rg int i=head[x];i;i=E[i].nx)
{
rg int y=E[i].to;
if(y==f)
continue;
dfs1(y,x);
siz[x]+=siz[y];
if(siz[y]>siz[son[x]])
son[x]=y;
}
}
int top[MAXN];
int dfn[MAXN],clk;
il void dfs2(rg int x,rg int tp)
{
top[x]=tp;
dfn[x]=++clk;
if(!son[x])
return;
dfs2(son[x],tp);
for(rg int i=head[x];i;i=E[i].nx)
{
rg int y=E[i].to;
if(y==fa[x]||y==son[x]) // edit 1
continue;
dfs2(y,y);
}
}
il int lca(rg int x,rg int y)
{
while(top[x]!=top[y])
{
if(dep[top[x]]<dep[top[y]])
swap(x,y);
x=fa[top[x]];
}
return dep[x]<dep[y]?x:y;
}
struct Quiz
{
int s,t,lca;
}Q[MAXN];
int ans[MAXN];
int root[MAXN*3],tcnt;
int ql,qr,v;
struct SegTree
{
int sumv[MAXN*20];
int L[MAXN*20],R[MAXN*20];
il void add(rg int&now,rg int l,rg int r)
{
if(!now)
now=++tcnt;
sumv[now]+=v;
if(l==r)
return;
rg int mid=(l+r)>>1;
if(ql<=mid)
add(L[now],l,mid);
else
add(R[now],mid+1,r);
}
il int qsum(rg int now,rg int l,rg int r)
{
if(!now)
return 0;
if(ql<=l&&r<=qr)
return sumv[now];
rg int mid=(l+r)>>1;
if(qr<=mid)
return qsum(L[now],l,mid);
if(ql>=mid+1)
return qsum(R[now],mid+1,r);
return qsum(L[now],l,mid)+qsum(R[now],mid+1,r);
}
}T;
il void init()
{
tcnt=0;
memset(root,0,sizeof root);
memset(&T,0,sizeof T);
}
int main()
{
// freopen(".in","r",stdin);
// freopen(".out","w",stdout);
rg int n,m;
read(n);read(m);
for(rg int i=1,x,y;i<n;++i)
{
read(x);read(y);
addedge(x,y);
addedge(y,x);
}
for(rg int i=1;i<=n;++i)
read(w[i]);
dfs1(1,0);
dfs2(1,1);
for(rg int i=1;i<=m;++i)
{
read(Q[i].s);read(Q[i].t);
Q[i].lca=lca(Q[i].s,Q[i].t);
// cerr<<i<<" lca="<<Q[i].lca<<endl;
}
for(rg int i=1;i<=n;++i)
t[i]=w[i]+dep[i];
for(rg int i=1,now;i<=m;++i)
{
now=dep[Q[i].s];
ql=dfn[Q[i].s],v=1;
T.add(root[now],1,n);
ql=dfn[fa[Q[i].lca]],v=-1; // 此处计算lca
if(ql) // edit 2:加0会加到1上
T.add(root[now],1,n);
}
for(rg int i=1;i<=n;++i)
{
ql=dfn[i],qr=dfn[i]+siz[i]-1;
ans[i]+=T.qsum(root[t[i]],1,n);
// cerr<<i<<" ans="<<ans[i]<<endl;
}
init();
for(rg int i=1;i<=n;++i)
t[i]=w[i]-dep[i]+n+1;
for(rg int i=1,now;i<=m;++i)
{
now=dep[Q[i].s]-2*dep[Q[i].lca]+n+1;
ql=dfn[Q[i].t],v=1;
T.add(root[now],1,n);
ql=dfn[Q[i].lca],v=-1; // 此处不计算lca
T.add(root[now],1,n);
}
for(rg int i=1;i<=n;++i)
{
ql=dfn[i],qr=dfn[i]+siz[i]-1;
ans[i]+=T.qsum(root[t[i]],1,n);
// cerr<<i<<" ans="<<ans[i]<<endl;
}
for(rg int i=1;i<=n;++i)
printf("%d ",ans[i]);
// fclose(stdin);
// fclose(stdout);
return 0;
}
LG1600 天天爱跑步的更多相关文章
- 【LG1600】[NOIP2016]天天爱跑步
[LG1600][NOIP2016]天天爱跑步 题面 洛谷 题解 考虑一条路径\(S\rightarrow T\)是如何给一个观测点\(x\)造成贡献的, 一种是从\(x\)的子树内出来,另外一种是从 ...
- UOJ261 【NOIP2016】天天爱跑步
本文版权归ljh2000和博客园共有,欢迎转载,但须保留此声明,并给出原文链接,谢谢合作. 本文作者:ljh2000作者博客:http://www.cnblogs.com/ljh2000-jump/转 ...
- BZOJ4719 [Noip2016]天天爱跑步
本文版权归ljh2000和博客园共有,欢迎转载,但须保留此声明,并给出原文链接,谢谢合作. 本文作者:ljh2000作者博客:http://www.cnblogs.com/ljh2000-jump/转 ...
- noip2016天天爱跑步
题目描述 小c同学认为跑步非常有趣,于是决定制作一款叫做<天天爱跑步>的游戏.«天天爱跑步»是一个养成类游戏,需要玩家每天按时上线,完成打卡任务. 这个游戏的地图可以看作一一棵包含 个结点 ...
- bzoj 4719: [Noip2016]天天爱跑步
Description 小c同学认为跑步非常有趣,于是决定制作一款叫做<天天爱跑步>的游戏.?天天爱跑步?是一个养成类游戏,需要 玩家每天按时上线,完成打卡任务.这个游戏的地图可以看作一一 ...
- [NOIP]2016天天爱跑步
[NOIP]2016天天爱跑步 标签: LCA 树上差分 NOIP Description 小C同学认为跑步非常有趣,于是决定制作一款叫做<天天爱跑步>的游戏.<天天爱跑步>是 ...
- NOIP2016 天天爱跑步 80分暴力
https://www.luogu.org/problem/show?pid=1600 题目描述 小c同学认为跑步非常有趣,于是决定制作一款叫做<天天爱跑步>的游戏.«天天爱跑步»是一个养 ...
- [NOIp 2016]天天爱跑步
Description 小C同学认为跑步非常有趣,于是决定制作一款叫做<天天爱跑步>的游戏.<天天爱跑步>是一个养成类游戏,需要玩家每天按时上线,完成打卡任务. 这个游戏的地图 ...
- 【NOIP2016】【LCA】【树上差分】【史诗级难度】天天爱跑步
学弟不是说要出丧题吗>>所以我就研究了1天lca又研究了1天tj然后研究了一天天天爱跑步,终于写了出来.(最后的平均用时为240ms...比学弟快了1倍...) 题意:给你颗树,然后有m个 ...
随机推荐
- crond 的注意点
本来在控制台调试的好好的程序,一通过crond运行, 就执行不了. 一般需要注意: 1) 环境变量 当前目录 会变成用户的Home 2) /var/log/messages 可以看出 crond 调 ...
- HTML基础知识(w3school)
http://www.w3school.com.cn/tags/tag_meta.asp
- win10 WiFi 密码查询 命令
如果你笔记本有WiFi,正好你系统又是Win10 那么... netsh wlan show profile * key=clear
- Angular2 -- 生命周期
组件生命周期钩子 指令和组件的实例有一个生命周期:新建.更新和销毁. 每个接口都有唯一的一个钩子方法,它们的名字是由接口名加上 ng前缀构成的.比如,OnInit接口的钩子方法叫做ngOnInit. ...
- 页面跳转 Server.Transfer和 Response.Redirect的区别
1.Server.Transfer 用于把处理的控制权从一个页面转移到另一个页面,在转移的工程中没有离开服务器内部控件(如request,session等)保存的信息不变.因此你能从a页面跳转到b页面 ...
- 无法打开物理文件“xxxx.mdf”。操作系统错误 5:“5(拒绝访问)”
- spring cloud图形化dashboard是如何实现指标的收集展示的
spring cloud图形化dashboard是如何实现指标的收集展示的 1.dashboard图形化界面入口 http://localhost:10000/hystrix.stream 说明:端口 ...
- SQL TUNING——从近半小时到几十毫秒的一次优化
昨天,一个用户的现场人员打电话紧急求助,说他们的一个系统卡了,半天不出结果,严重的影响了他们的使用,我简单的问了几句:什么时候的事儿?答:就今天下午的事儿.问:数据库软硬件最近动过没?答:没动过.问: ...
- js获得焦点和失去焦点那些事
<!doctype html> <html> <head> <meta charset="utf-8"> <meta name ...
- SGU 124. Broken line 射线法 eps的精准运用,计算几何 难度:3
124. Broken line time limit per test: 0.25 sec. memory limit per test: 4096 KB There is a closed bro ...