BZOJ 4719 [Noip2016]天天爱跑步 ——树链剖分
一直以为自己当时是TLE了,但是再看发现居然WA?
然后把数组扩大一倍,就A掉了。QaQ
没什么好说的。一段路径分成两段考虑,上升的一段深度+时间是定值,下降的一段深度-时间是定值,然后打标记统计即可。
发现大概是统计数组因为深度+时间太大炸掉了。
现在想想,当时没有对拍,真是后怕。
#include <cstdio>
#include <vector>
#include <cstring>
#include <iostream>
#include <algorithm>
using namespace std;
int read()
{
int ret=0,f=1; char ch=getchar();
while (ch<'0'||ch>'9'){if (ch=='-') f=-1;ch=getchar();}
while (ch>='0'&&ch<='9'){ret*=10;ret+=ch-'0';ch=getchar();}
return ret*f;
}
int n,m;
int h[600005],to[600002],ne[600002],en=0;
void add(int a,int b)
{
to[en]=b;
ne[en]=h[a];
h[a]=en++;
}
int root=1,dep[600005],fa[600005],son[600005],siz[600005];
vector <int> st[600005];
vector <int> ed[600005];
void dfs1(int k)
{
siz[k]=1;
for (int i=h[k];i>=0;i=ne[i])
if (to[i]!=fa[k]){
fa[to[i]]=k;
dep[to[i]]=dep[k]+1;
dfs1(to[i]);
siz[k]+=siz[to[i]];
if (siz[son[k]]<siz[to[i]]) son[k]=to[i];
}
}
int pos[600005],top[600005],li[600005],cnt=0;
void dfs2(int k,int tp)
{
top[k]=tp;
cnt++; pos[k]=cnt; li[cnt]=k;
if (!son[k]) return ;
if (son[k]) dfs2(son[k],tp);
for (int i=h[k];i>=0;i=ne[i])
if (to[i]!=son[k]&&to[i]!=fa[k]) dfs2(to[i],to[i]);
return ;
}
int w[600005];
vector <int> que[600005];
vector <int> bel[600005];
struct node{int s,t,lca;}q[600005];
int vis[600005],f[600005];
int gf(int k)
{
if (f[k]==k) return k;
else return f[k]=gf(f[k]);
}
void dfs3(int k)
{
f[k]=k;
vis[k]=1;
for (int i=0;i<que[k].size();++i)
if (vis[que[k][i]]) q[bel[k][i]].lca=gf(que[k][i]);
for (int i=h[k];i>=0;i=ne[i])
if (fa[k]!=to[i]){
dfs3(to[i]);
f[to[i]]=k;
}
return ;
}
void lca(){dfs3(1);}
int s[600005],ans[600005];
void add(int a,int b,int fn)
{
if (dep[a]<dep[b]) swap(a,b);
while (top[a]!=top[b])
{
if (dep[top[a]]<dep[top[b]]) swap(a,b);
st[pos[top[a]]].push_back(fn);
ed[pos[a]+1].push_back(fn);
a=fa[top[a]];
}
if (dep[a]<dep[b]) swap(a,b);
st[pos[b]].push_back(fn);
ed[pos[a]+1].push_back(fn);
}
void cal1()
{
for (int i=1;i<=n;++i) w[i]-=dep[i];
for (int i=1;i<=m;++i)
{
int at;
at=dep[q[i].s]-dep[q[i].lca];
if (w[q[i].lca]==at-dep[q[i].lca]) ans[q[i].lca]--;
add(q[i].lca,q[i].t,at-dep[q[i].lca]);
}
for (int i=1;i<=n;++i)
{
for (int j=0;j<st[i].size();++j) s[st[i][j]]++;
for (int j=0;j<ed[i].size();++j) s[ed[i][j]]--;
ans[li[i]]+=s[w[li[i]]];
}
}
void cal2()
{
for (int i=0;i<=n+5;++i) st[i].clear(),ed[i].clear();
memset(s,0,sizeof s);
for (int i=1;i<=n;++i) w[i]+=2*dep[i];
for (int i=1;i<=m;++i)
{
int at;
at=dep[q[i].s]-dep[q[i].lca];
add(q[i].lca,q[i].s,at+dep[q[i].lca]);
}
for (int i=1;i<=n;++i)
{
for (int j=0;j<st[i].size();++j) s[st[i][j]]++;
for (int j=0;j<ed[i].size();++j) s[ed[i][j]]--;
ans[li[i]]+=s[w[li[i]]];
}
}
void out()
{
for (int i=1;i<n;++i)printf("%d ",ans[i]); printf("%d",ans[n]);
}
int main()
{
memset(h,-1,sizeof h);
n=read(); m=read();
for (int i=1;i<n;++i)
{
int a,b;
a=read(); b=read();
add(a,b); add(b,a);
}
for (int i=1;i<=n;++i)
w[i]=read();
for (int i=1;i<=m;++i)
{
int a,b;
a=read(); b=read();
que[a].push_back(b);
que[b].push_back(a);
bel[a].push_back(i);
bel[b].push_back(i);
q[i].s=a;q[i].t=b;
}
dfs1(root);
dfs2(root,root);
lca();
cal1();
cal2();
out();
return 0;
}
BZOJ 4719 [Noip2016]天天爱跑步 ——树链剖分的更多相关文章
- NOIP2016提高组Day1T2 天天爱跑步 树链剖分 LCA 倍增 差分
原文链接https://www.cnblogs.com/zhouzhendong/p/9275606.html 题目传送门 - 洛谷P1600 题目传送门 - LOJ#2359 题目传送门 - Vij ...
- bzoj 4719: [Noip2016]天天爱跑步
Description 小c同学认为跑步非常有趣,于是决定制作一款叫做<天天爱跑步>的游戏.?天天爱跑步?是一个养成类游戏,需要 玩家每天按时上线,完成打卡任务.这个游戏的地图可以看作一一 ...
- bzoj 4719: [Noip2016]天天爱跑步【树上差分+dfs】
长久以来的心理阴影?但是其实非常简单-- 预处理出deep和每组st的lca,在这里我简单粗暴的拿树剖爆算了 然后考虑对于一组s t lca来说,被这组贡献的观察员x当且仅当: x在s到lca的路径上 ...
- [BZOJ 2243] [SDOI 2011] 染色 【树链剖分】
题目链接:BZOJ - 2243 题目分析 树链剖分...写了200+行...Debug了整整一天+... 静态读代码读了 5 遍 ,没发现错误,自己做小数据也过了. 提交之后全 WA . ————— ...
- BZOJ 4568 [Scoi2016]幸运数字(树链剖分 + 异或线性基)
题目链接 BZOJ 4568 考虑树链剖分+线段树维护每一段区域的异或线性基 对于每个询问,求出该点集的异或线性基.然后求一下这个线性基里面能异或出的最大值即可. #include <bits ...
- Bzoj 4196: [Noi2015]软件包管理器 树链剖分
4196: [Noi2015]软件包管理器 Time Limit: 10 Sec Memory Limit: 512 MBSubmit: 721 Solved: 419[Submit][Statu ...
- BZOJ 1969: [Ahoi2005]LANE 航线规划( 树链剖分 )
首先我们要时光倒流, 倒着做, 变成加边操作维护关键边. 先随意搞出一颗树, 树上每条边都是关键边(因为是树, 去掉就不连通了)....然后加边(u, v)时, 路径(u, v)上的所有边都变成非关键 ...
- bzoj 4196 [Noi2015]软件包管理器 (树链剖分+线段树)
4196: [Noi2015]软件包管理器 Time Limit: 10 Sec Memory Limit: 512 MBSubmit: 2852 Solved: 1668[Submit][Sta ...
- BZOJ - 3631 松鼠的新家 (树链剖分)
题目链接 树链剖分基础题,路径权值修改+差分 #include<bits/stdc++.h> using namespace std; typedef long long ll; ; in ...
随机推荐
- 解决spring http输入流和输出流只能读取一次
1.需求:给某些请求接口记录日志,记录请求的数据和响应的数据和请求所花费的时间.这里采用非侵入式编程,也业务代码进行解耦.按照spring AOP 的编程思想. 2.编程设计:在spring 拦截器中 ...
- c++的const和static区别
const定义的常量在超出其作用域之后其空间会被释放,而static定义的静态常量在函数执行后不会释放其存储空间. static表示的是静态的.类的静态成员函数.静态成员变量是和类相关的,而不是和类的 ...
- Android学习总结(十三) ———— ListView 简单用法
一.ListView的基本概念 在Android所有常用的原生控件当中,用法最复杂的应该就是ListView了,它专门用于处理那种内容元素很多,手机屏幕无法展示出所有内容的情况.ListView可以使 ...
- windows 操作系统种类
@hcy 敬请访问:http://blog.sina.com.cn/iihcy Microsoft公司从1983年开始研制Windows系统,最初的研制目标是在MS-DOS的基础上提供一个多任务的图形 ...
- 洛谷 P1311 选择客栈
题目描述 丽江河边有n 家很有特色的客栈,客栈按照其位置顺序从 1 到n 编号.每家客栈都按照某一种色调进行装饰(总共 k 种,用整数 0 ~ k-1 表示),且每家客栈都设有一家咖啡店,每家咖啡店均 ...
- Android(java)学习笔记145:Handler消息机制的原理和实现
联合学习 Android 异步消息处理机制 让你深入理解 Looper.Handler.Message三者关系 1. 首先我们通过一个实例案例来引出一个异常: (1)布局文件activity_m ...
- win7上安装虚拟机并上网
一.安装Workstation Pro 二.下载CentOS-7-x86_64-DVD-1511.iso包 三.创建新的虚拟机,按照向导安装即可 四.使用cd /etc/sysconfig/netwo ...
- UEditor1.4.3的实例程序
官网:http://ueditor.baidu.com/website/ 配置下就可以使用 (1)下载,解压后文件结构如下: (2)将整个文件夹改名ueditor后复制到WebRoot目录下: (3) ...
- app内嵌H5调用分享
最近产品提出了一个需求:我们在合作方的app中提供的部分页面中增加分享页面,具体要求是在3个二维码推广页面调用app的分享接口,分享方式有3种,分别是点击”分享链接“按钮调起分享,点击”分享图片“按钮 ...
- NOIP 2017 图书管理员
题目描述 图书馆中每本书都有一个图书编码,可以用于快速检索图书,这个图书编码是一个 正整数. 每位借书的读者手中有一个需求码,这个需求码也是一个正整数.如果一本书的图 书编码恰好以读者的需求码结尾,那 ...