bzoj 4719: [Noip2016]天天爱跑步
Description
Input
Output
输出1行N 个整数,第个整数表示结点的观察员可以观察到多少人。
Sample Input
2 3
1 2
1 4
4 5
4 6
0 2 5 1 2 3
1 5
1 3
2 6
Sample Output
HINT
Source
昨天终于把这个史前巨坑给填了
对于一个点能否产生贡献我们分两种(拆路径)情况讨论:
1.在s--lca间,那么T[i]=deep[s]-deep[i],移项:T[i]+deep[i]=deep[s];
1.在lca---t之间,那么T[i]=deep[s]+deep[i]-2*deep[lca],移项:deep[i]-T[i]=2*deep[lca]-deep[s];
我们发现等式右边只和i有关,我们相当于要把路径上某些满足条件的点加1
于是好像用路径分块+桶可以过掉95分,(n*sqrt(n)*logn);
正解是用了一种类似于树上差分打标记的方法(在s和t加,lca处减掉),具体思想比较巧妙,用差分后相当于查询子树和(查询桶中特定值),具体实现不太好讲,代码比较清楚。
分块:
// MADE BY QT666
#include<cstdio>
#include<algorithm>
#include<cmath>
#include<iostream>
#include<cstring>
#define RG register
using namespace std;
typedef long long ll;
const int N=100050;
int gi()
{
int x=0,flag=1;
char ch=getchar();
while(ch<'0'||ch>'9'){if(ch=='-') flag=-1;ch=getchar();}
while(ch>='0'&&ch<='9') x=x*10+ch-'0',ch=getchar();
return x*flag;
}
int head[N*2],to[N*2],nxt[N*2];
int deep[N],size[N],son[N],top[N],fa[N],cnt,T[N],dfn[N],pd[2][N];
int n,m,sum,pos[N],block,L[N],R[N],tt,id[N];
int tong[320][N*2][2],ans[N],tot;
struct data{
int l,r,flag;
}p[N];
inline void lnk(int x,int y){
to[++cnt]=y,nxt[cnt]=head[x],head[x]=cnt;
to[++cnt]=x,nxt[cnt]=head[y],head[y]=cnt;
}
inline void dfs1(RG int x,RG int f){
deep[x]=deep[f]+1;size[x]=1;
for(RG int i=head[x];i;i=nxt[i]){
int y=to[i];
if(y!=f){
fa[y]=x;dfs1(y,x);
size[x]+=size[y];
if(size[y]>size[son[x]]) son[x]=y;
}
}
}
inline void dfs2(RG int x,RG int f){
dfn[x]=++sum;id[sum]=x;top[x]=f;
if(son[x]) dfs2(son[x],f);
for(RG int i=head[x];i;i=nxt[i]){
int y=to[i];
if(y!=fa[x]&&y!=son[x]) dfs2(y,y);
}
}
inline int LCA(RG int x,RG int y){
tot=0;int fl=1;
while(top[x]!=top[y]){
if(deep[top[x]]<deep[top[y]]) swap(x,y),fl^=1;
p[++tot]=(data){dfn[top[x]],dfn[x],fl};
x=fa[top[x]];
}
if(deep[x]<deep[y]) swap(x,y),fl^=1;
p[++tot]=(data){dfn[y],dfn[x],fl};
return y;
}
inline void update(RG int l,RG int r,RG int x,int flag){
if(pos[l]==pos[r]){
for(RG int i=l;i<=r;i++){
if(pd[flag][id[i]]==x) ans[id[i]]++;
}
}
else{
for(RG int i=pos[l]+1;i<pos[r];i++) tong[i][x][flag]++;
for(RG int i=l;i<=R[pos[l]];i++){
if(pd[flag][id[i]]==x) ans[id[i]]++;
}
for(RG int i=L[pos[r]];i<=r;i++){
if(pd[flag][id[i]]==x) ans[id[i]]++;
}
}
}
int main(){
freopen("running.in","r",stdin);
freopen("running.out","w",stdout);
n=gi(),m=gi();
for(RG int i=1;i<n;i++){
int x=gi(),y=gi();lnk(x,y);
}
dfs1(1,0),dfs2(1,1);
block=sqrt(n);tt=n/block;
if(n%block) tt++;
for(RG int i=1;i<=n;i++) pos[i]=(i-1)/block+1;
for(RG int i=1;i<=tt;i++) L[i]=(i-1)*block+1,R[i]=i*block;
R[tt]=n;
for(RG int i=1;i<=n;i++) T[i]=gi(),pd[0][i]=deep[i]-T[i]+N,pd[1][i]=deep[i]+T[i]+N;
for(RG int i=1;i<=m;i++){
int s=gi(),t=gi(),lca=LCA(s,t);
for(RG int j=1;j<=tot;j++){
if(p[j].flag==1) update(p[j].l,p[j].r,deep[s]+N,1);
else update(p[j].l,p[j].r,2*deep[lca]-deep[s]+N,0);
}
}
for(RG int i=1;i<=n;i++){
printf("%d ",ans[i]+tong[pos[dfn[i]]][pd[0][i]][0]+tong[pos[dfn[i]]][pd[1][i]][1]);
}
}
正解:
// MADE BY QT666
#include<cstdio>
#include<algorithm>
#include<cmath>
#include<iostream>
#include<vector>
#include<cstring>
#define RG register
using namespace std;
typedef long long ll;
const int N=500050;
int gi()
{
int x=0,flag=1;
char ch=getchar();
while(ch<'0'||ch>'9'){if(ch=='-') flag=-1;ch=getchar();}
while(ch>='0'&&ch<='9') x=x*10+ch-'0',ch=getchar();
return x*flag;
}
int head[N*2],to[N*2],nxt[N*2];
int deep[N],size[N],son[N],top[N],fa[N],cnt,T[N],dfn[N];
int n,m,sum,tt,id[N],ans[N],maxn;
int tong[2*N],tong2[2*N],num[N];
vector<int>a[N],b[N],c[N];
struct data{
int s,t,lca,len;
}q[N];
inline void lnk(int x,int y){
to[++cnt]=y,nxt[cnt]=head[x],head[x]=cnt;
to[++cnt]=x,nxt[cnt]=head[y],head[y]=cnt;
}
inline void dfs1(RG int x,RG int f){
deep[x]=deep[f]+1;size[x]=1;
maxn=max(deep[x],maxn);
for(RG int i=head[x];i;i=nxt[i]){
int y=to[i];
if(y!=f){
fa[y]=x;dfs1(y,x);
size[x]+=size[y];
if(size[y]>size[son[x]]) son[x]=y;
}
}
}
inline void dfs2(RG int x,RG int f){
dfn[x]=++sum;id[sum]=x;top[x]=f;
if(son[x]) dfs2(son[x],f);
for(RG int i=head[x];i;i=nxt[i]){
int y=to[i];
if(y!=fa[x]&&y!=son[x]) dfs2(y,y);
}
}
inline int LCA(RG int x,RG int y){
while(top[x]!=top[y]){
if(deep[top[x]]<deep[top[y]]) swap(x,y);
x=fa[top[x]];
}
if(deep[x]<deep[y]) swap(x,y);
return y;
}
inline void DFS_up(int x,int f){
int now=T[x]+deep[x],last=tong[now+N];
for(int i=head[x];i;i=nxt[i]) {
int y=to[i];
if(y!=f) DFS_up(y,x);
}
tong[deep[x]+N]+=num[x];
ans[x]=tong[now+N]-last;
for(int i=0;i<a[x].size();i++) tong[deep[a[x][i]]+N]--;
} inline void DFS_down(int x,int f){
int now=deep[x]-T[x],last=tong2[now+N];
for(int i=head[x];i;i=nxt[i]) {
int y=to[i];
if(y!=f) DFS_down(y,x);
}
for(int i=0;i<b[x].size();i++) tong2[N+b[x][i]]++;
ans[x]+=tong2[now+N]-last;
for(int i=0;i<c[x].size();i++) tong2[N+c[x][i]]--;
}
int main(){
freopen("running.in","r",stdin);
freopen("running.out","w",stdout);
n=gi(),m=gi();
for(RG int i=1;i<n;i++){
int x=gi(),y=gi();lnk(x,y);
}
dfs1(1,0),dfs2(1,1);
for(int i=1;i<=n;i++) T[i]=gi();
for(int i=1;i<=m;i++){
q[i].s=gi(),q[i].t=gi();num[q[i].s]++;
q[i].lca=LCA(q[i].s,q[i].t);
q[i].len=deep[q[i].s]+deep[q[i].t]-2*deep[q[i].lca];
a[q[i].lca].push_back(q[i].s);
}
DFS_up(1,0);
for(int i=1;i<=m;i++){
b[q[i].t].push_back(deep[q[i].t]-q[i].len);
c[q[i].lca].push_back(deep[q[i].t]-q[i].len);
}
DFS_down(1,0);
for(int i=1;i<=m;i++) if(deep[q[i].s]-deep[q[i].lca]==T[q[i].lca]) ans[q[i].lca]--;
printf("%d",ans[1]);
for(int i=2;i<=n;i++) printf(" %d",ans[i]);
}
bzoj 4719: [Noip2016]天天爱跑步的更多相关文章
- BZOJ 4719 [Noip2016]天天爱跑步 ——树链剖分
一直以为自己当时是TLE了,但是再看发现居然WA? 然后把数组扩大一倍,就A掉了.QaQ 没什么好说的.一段路径分成两段考虑,上升的一段深度+时间是定值,下降的一段深度-时间是定值,然后打标记统计即可 ...
- bzoj 4719: [Noip2016]天天爱跑步【树上差分+dfs】
长久以来的心理阴影?但是其实非常简单-- 预处理出deep和每组st的lca,在这里我简单粗暴的拿树剖爆算了 然后考虑对于一组s t lca来说,被这组贡献的观察员x当且仅当: x在s到lca的路径上 ...
- 4719: [Noip2016]天天爱跑步
Time Limit: 40 Sec Memory Limit: 512 MB Submit: 1986 Solved: 752 [Submit][Status][Discuss] Descripti ...
- [NOIp2016]天天爱跑步 线段树合并
[NOIp2016]天天爱跑步 LG传送门 作为一道被毒瘤出题人们玩坏了的NOIp经典题,我们先不看毒瘤的"动态爱跑步"和"天天爱仙人掌",回归一下本来的味道. ...
- [Noip2016]天天爱跑步 LCA+DFS
[Noip2016]天天爱跑步 Description 小c同学认为跑步非常有趣,于是决定制作一款叫做<天天爱跑步>的游戏.?天天爱跑步?是一个养成类游戏,需要玩家每天按时上线,完成打卡任 ...
- 【LG1600】[NOIP2016]天天爱跑步
[LG1600][NOIP2016]天天爱跑步 题面 洛谷 题解 考虑一条路径\(S\rightarrow T\)是如何给一个观测点\(x\)造成贡献的, 一种是从\(x\)的子树内出来,另外一种是从 ...
- NOIP2016天天爱跑步 题解报告【lca+树上统计(桶)】
题目描述 小c同学认为跑步非常有趣,于是决定制作一款叫做<天天爱跑步>的游戏.«天天爱跑步»是一个养成类游戏,需要玩家每天按时上线,完成打卡任务. 这个游戏的地图可以看作一一棵包含 nn个 ...
- BZOJ4719 [Noip2016]天天爱跑步
本文版权归ljh2000和博客园共有,欢迎转载,但须保留此声明,并给出原文链接,谢谢合作. 本文作者:ljh2000作者博客:http://www.cnblogs.com/ljh2000-jump/转 ...
- noip2016天天爱跑步
题目描述 小c同学认为跑步非常有趣,于是决定制作一款叫做<天天爱跑步>的游戏.«天天爱跑步»是一个养成类游戏,需要玩家每天按时上线,完成打卡任务. 这个游戏的地图可以看作一一棵包含 个结点 ...
随机推荐
- 3255:十进制到六进制-poj
3255:十进制到六进制 总时间限制: 1000ms 内存限制: 65536kB 描述 进制转换: 将十进制(不超过int类型表示的范围)的数转换为六进制的数. 输入 输入为第一行是组数n,后面n ...
- StringUtils工具类常用方法汇总2(截取、去除空白、包含、查询索引)
在上一篇中总结了StringUtils工具类在判断字符串为空,大小写转换,移除字符或字符序列,替换,反转,切割合并等方面的方法,这次再汇总一下其它常用的方法. 一.截取 StringUtils ...
- 爬起点小说day03
# 把所有类别的前3页的小说爬取下来 import scrapyfrom scrapy.http import Requestfrom time import sleepfrom qidianNove ...
- 基于 HTML5 Canvas 的 3D 碰撞检测
这是公司大神写的一个放官网上给用户学习的例子,我一开始真的不知道这是在干嘛,就只是将三个形状图元组合在一起,然后可以同时旋转.放大缩小这个三个图形,点击"Animate"就能让中间 ...
- php加密解密处理类
[PHP]代码 <?php /*=========================================================== = 版权协议: = GPL (The GN ...
- SQL Server学习之路(七):Python3操作SQL Server数据库
0.目录 1.前言 2.准备工作 3.简单测试语句 4.提交与回滚 5.封装成类的写法 1.前言 前面学完了SQL Server的基本语法,接下来学习如何在程序中使用sql,毕竟不能在程序中使用的话, ...
- web前端素材整理汇总
最近一直搞前端开发,整理下前端用的一些常用素材,分享给大家 框架类 Vue:https://cn.vuejs.org/ iview:https://www.iviewui.com/ 插件类 Jquer ...
- codeforge免费下载账号 积分账号 共享账号
codeforge网站下载代码很好,没有积分怎么办?那么多好的matlab代码,matlab程序,C,JAVA等等,都要充值啊!!! 请用下面共享的codeforge账号密码========> ...
- hibernate5(9)注解映射[1]多对一单向关联
在博客站点中,我们可能须要从某一篇文章找到其所关联的作者.这就须要从文章方建立起对用户的关联,即是多对一的映射关系. 如今先看一个配置实例:我们的文章实体类 package com.zeng.mode ...
- 单点登录(一)使用Cookie+File实现单点登录
本文使用Cookies+Filter实现www.taobao.tgb.com 和 www.tianmao.tgb.com的单点登录. 源代码分享:链接: http://pan.baidu.com/s/ ...