https://www.luogu.org/problemnew/show/P1600

https://www.zybuluo.com/wsndy-xx/note/1135243

乱写的暴力,这道题暴力写个60还是比较简单的

#include <iostream>
#include <cstdio>
#include <algorithm>
#include <queue>
#include <map>
#include <vector>
#include <cstring> using namespace std;
const int N = ;
const int oo = ; #define lson jd << 1
#define rson jd << 1 | 1 #define yxy getchar()
#define one_ n <= 993
#define two_ n == 99994
#define three_ n == 99995 int head[N], pre[N], dis[N], tim[N], Answer[N], Askl[N], Askr[N];
bool vis[N];
int n, m, now = ;
struct Node {
int u, v, w, nxt;
} G[N];
queue <int> Q;
vector <int> Vt[N];
int L[N << ], R[N << ], W[N << ], F[N << ]; inline int read() {
int x = ;
char c = yxy;
while(c < '' || c > '') c = yxy;
while(c >= '' && c <= '') x = x * + c - '', c = yxy;
return x;
} inline void add(int u, int v) {
G[now].v = v;
G[now].nxt = head[u];
head[u] = now ++;
} inline void spfa(int start,int endd) {
for(int i = ; i <= n; i ++) dis[i] = oo, vis[i] = ;
dis[start] = ;
Q.push(start);
while(!Q.empty()) {
int topp = Q.front();
Q.pop();
vis[topp] = ;
for(int i = head[topp]; ~ i; i = G[i].nxt) {
if(dis[G[i].v] > dis[topp] + ) {
dis[G[i].v] = dis[topp] + ;
pre[G[i].v] = topp;
if(!vis[G[i].v]) {
vis[G[i].v] = ;
Q.push(G[i].v);
}
}
}
}
} void calc(int start, int endd, int diss) {
int js = -;
pre[start] = ;
while(endd) {
js ++;
if(tim[endd] == diss - js) Answer[endd] ++;
endd = pre[endd];
}
} void work_1() {
for(int i = ; i <= m; i ++) {
spfa(Askl[i], Askr[i]);
calc(Askl[i], Askr[i], dis[Askr[i]]);
}
} void work_2() {
for(int i = ; i <= m; i ++) Vt[Askl[i]].push_back(Askr[i]);
for(int i = ; i <= n; i ++) {
int L_ = i - tim[i], R_ = i + tim[i];
int siz_ = Vt[L_].size();
if(L_ > )
for(int j = ; j < siz_; j ++)
if(Vt[L_][j] >= i) Answer[i] ++;
siz_ = Vt[R_].size();
if(R_ <= n)
for(int j = ; j < siz_; j ++)
if(Vt[R_][j] <= i) Answer[i] ++;
}
} int bef[N], top[N], deep[N], size[N], fa[N], son[N], tree[N], spjs;
int cnt[N]; void Dfs_3(int u, int f_, int dep) {
fa[u] = f_, deep[u] = dep; size[u] = ;
for(int i = head[u]; ~ i; i = G[i].nxt) {
int v = G[i].v;
if(v != f_) {
dis[v] = dis[u] + ;
Dfs_3(v, u, dep + );
size[u] += size[v];
cnt[u] += cnt[v];
}
}
} inline void work_3() {
memset(dis, , sizeof dis);
for(int i = ; i <= m; i ++) cnt[Askr[i]] ++;
Dfs_3(, , );
for(int i = ; i <= n; i ++)
if(deep[i] == tim[i])
Answer[i] += cnt[i];
} int main() {
n = read();
m = read();
for(int i = ; i <= n; i ++) head[i] = -;
for(int i = ; i <= n - ; i ++) {
int u = read();
int v = read();
add(u, v);
add(v, u);
}
for(int i = ; i <= n; i ++) tim[i] = read();
for(int i = ; i <= m; i ++) Askl[i] = read(), Askr[i] = read();
if(one_) work_1();
else if(two_) work_2();
else if(three_) work_3(); //起点 == 1
for(int i = ; i <= n; i ++) printf("%d ", Answer[i]);
return ;
}

前置知识

Lca + 线段树 + 差分 + 树剖

考虑把一条路径拆成两段(这是非常常见的解决树上问题的方法)

分别拆成 S - L 和 L - T (起点 -  Lca, Lca  -  终点)

这样就可以得到当满足

deep[s] - deep[i] = wat[i]  =>  deep[s] = wat[i] + deep[i];

deep[s] + deep[i] - 2 * deep[Lca(s, i)] = wat[i]  =>  deep[s] - 2 * deep[Lca(s, i)] = wat[i] - deep[i];

时玩家才会被 i 观察员看到

发现 上面两个式子满足等式右边都是定值

因此我们可以 以深度建立线段树(动态开节点)

查询时就应该查询该节点所对应的深度的线段树的区间lst[] 和 rst[] 之间的总的权值

lst[i] 表示以该节点为子树的根中树上编号的下界, 同理rst[]为上界(涉及到DFS序 && 树剖的知识).

#include <iostream>
#include <cstdio>
#include <cstring> using namespace std;
const int N = 3e5 + ; #define gc getchar() int n, m, wat[N];
int Askl[N], Askr[N], Lca[N]; int now = , head[N];
struct Node {int v, nxt;} G[N << ]; inline int read() {
int x = ; char c = gc;
while(c < '' || c > '') c = gc;
while(c >= '' && c <= '') x = x * + c - '', c = gc;
return x;
} inline void Add(int u, int v) {G[now].v = v; G[now].nxt = head[u]; head[u] = now ++;} int fa[N], deep[N], top[N], size[N], son[N], lst[N], rst[N], tree[N], Spjs; void Dfs_1(int u, int f_, int dep) {
fa[u] = f_; deep[u] = dep; size[u] = ;
for(int i = head[u]; ~ i; i = G[i].nxt) {
int v = G[i].v;
if(v != f_) {
Dfs_1(v, u, dep + );
size[u] += size[v];
if(size[son[u]] < size[v]) son[u] = v;
}
}
} void Dfs_2(int u, int tp) {
top[u] = tp;
lst[u] = ++ Spjs;
tree[u] = Spjs;
if(! son[u]) {rst[u] = Spjs; return ;}
Dfs_2(son[u], tp);
for(int i = head[u]; ~ i; i = G[i].nxt) {
int v = G[i].v;
if(v != fa[u] && v != son[u]) Dfs_2(v, v);
}
rst[u] = Spjs;
} inline int Ask_Lca(int x, int y) {
int tp1 = top[x], tp2 = top[y];
while(tp1 != tp2) {
if(deep[tp1] < deep[tp2]) swap(x, y), swap(tp1, tp2);
x = fa[tp1];
tp1 = top[x];
}
return deep[x] < deep[y] ? x : y;
} int root[N * ], lson[N * ], rson[N * ], W[N * ], tot, cnt; void Build_G(int l, int r, int & jd, int x, int yj) {
if(!x) return ;
if(!jd) jd = ++ tot;
W[jd] += yj;
if(l == r) return ;
int mid = (l + r) >> ;
if(x <= mid) Build_G(l, mid, lson[jd], x, yj);
else Build_G(mid + , r, rson[jd], x, yj);
} int Sec_A(int jd, int l, int r, int x, int y) {
if(! jd) return ;
if(x <= l && r <= y) return W[jd];
int mid = (l + r) >> ;
if(y <= mid) return Sec_A(lson[jd], l, mid, x, y);
else if(x > mid) return Sec_A(rson[jd], mid + , r, x, y);
else return Sec_A(lson[jd], l, mid, x, y) + Sec_A(rson[jd], mid + , r, x, y);
} void Clear() {
tot = ;
memset(lson, , sizeof lson);
memset(rson, , sizeof rson);
memset(W, , sizeof W);
memset(root, , sizeof root);
} int Answer[N]; int main() {
n = read(); m = read();
for(int i = ; i <= n; i ++) head[i] = -;
for(int i = ; i <= n - ; i ++) {
int u = read(), v = read();
Add(u, v); Add(v, u);
}
for(int i = ; i <= n; i ++) wat[i] = read();
for(int i = ; i <= m; i ++) Askl[i] = read(), Askr[i] = read();
Dfs_1(, , );
Dfs_2(, );
for(int i = ; i <= m; i ++) Lca[i] = Ask_Lca(Askl[i], Askr[i]);
int dep;
for(int i = ; i <= m; i ++) {
dep = deep[Askl[i]];
Build_G(, n, root[dep], tree[Askl[i]], );
Build_G(, n, root[dep], tree[fa[Lca[i]]], -);
}
for(int i = ; i <= n; i ++)
Answer[i] = Sec_A(root[deep[i] + wat[i]], , n, lst[i], rst[i]);
Clear();
for(int i = ; i <= m; i ++) {
dep = deep[Askl[i]] - deep[Lca[i]] * + n * ;
Build_G(, n, root[dep], tree[Askr[i]], );
Build_G(, n, root[dep], tree[Lca[i]], -);
}
for(int i = ; i <= n; i ++)
Answer[i] += Sec_A(root[wat[i] - deep[i] + n * ], , n, lst[i], rst[i]);
for(int i = ; i <= n; i ++)
cout << Answer[i] << " ";
return ;
}

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

  1. [luogu]P1600 天天爱跑步[LCA]

    [luogu]P1600 [NOIP 2016]天天爱跑步 题目描述 小c同学认为跑步非常有趣,于是决定制作一款叫做<天天爱跑步>的游戏.«天天爱跑步»是一个养成类游戏,需要玩家每天按时上 ...

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

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

  3. Luogu P1600[NOIP2016]day1 T2天天爱跑步

    号称是noip2016最恶心的题 基本上用了一天来搞明白+给sy讲明白(可能还没讲明白 具体思路是真的不想写了(快吐了 如果要看,参见洛谷P1600 天天爱跑步--题解 虽然这样不好但我真的不想写了 ...

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

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

  5. luoguP1600 天天爱跑步(NOIP2016)(主席树+树链剖分)

    阅读体验: https://zybuluo.com/Junlier/note/1303550 为什么这一篇的Markdown炸了? # 天天爱跑步题解(Noip2016)(桶+树上差分 ^ 树剖+主席 ...

  6. UOJ261 【NOIP2016】天天爱跑步

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

  7. BZOJ4719 [Noip2016]天天爱跑步

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

  8. noip2016天天爱跑步

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

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

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

随机推荐

  1. sshpass ssh登录时自动输入密码

    安装 直接安装 sudo apt-get install sshpass 源代码安装 wget http://sourceforge.net/projects/sshpass/files/sshpas ...

  2. 剑指offer49:把字符串转换成整数

    1 题目描述 将一个字符串转换成一个整数(实现Integer.valueOf(string)的功能,但是string不符合数字要求时返回0),要求不能使用字符串转换整数的库函数. 数值为0或者字符串不 ...

  3. VS2017的一些调试方法技巧

    一.基本的操作. 1.启动调试. 可以通过VS的调试(Debug)菜单启动调试.点击调试菜单下的“启动调试”或者按F5键启动.如果你已经在代码中加入了断点,那么执行会自动开始. 注:退出调试快捷键sh ...

  4. Elastic Search快速上手(3):搜索

    前言 存储好数据之后,便可通过RESTful API进行搜索. 详细文档可参考: --简单搜索https://www.elastic.co/guide/cn/elasticsearch/guide/c ...

  5. Spring实战(二)Spring容器和bean的生命周期

    引入问题: 在XML配置文件中配置bean后,这些文件又是如何被加载的?它们被加载到哪里去了? Spring容器——框架核心 1.什么是Spring容器?它的功能是什么? 在基于Spring的应用中, ...

  6. Java组合模式(思维导图)

    图1 组合模式[点击查看图片] 1,以公司职员为例的结构 package com.cnblogs.mufasa.demo3; import java.util.ArrayList; import ja ...

  7. Windows phone 8 二维码生成与扫描

    1. 二维码的生成 二维码生成用到了一个第三方的插件(zxing.wp8.0) 根据指定的信息,生成对应的二维码. 代码很简单: bool falg=tbk.Text==""?fa ...

  8. 解决 Oracle TNSListener 服务启动找不到路径问题

    TNSListener服务无法启动,提示从系统无法找到指定路径! 解决方法: 在控制面板/管理工具/服务中双击打开OracleOraHome92TNSListener的服务看到其 “可执行文件的路径” ...

  9. express相关操作

    1.   安装应用生成器npm install express-generator –g 2.   生成项目Express --view=ejs myapp 3.   进入项目安装依赖包 npm in ...

  10. 使用代码将github仓库里某个issue同步到CSDN博客上

    我是一个懒惰的程序员.我在github仓库里用issue的方式写了很多分享文章,想同步到CSDN上.但是我又不想一篇篇手动复制粘贴,因此想用代码来实现自动化. 例子: https://github.c ...