@bzoj - 4381@ [POI2015] Odwiedziny
@description@
给定一棵 n 个点的树,树上每条边的长度都为 1 ,第 i 个点的权值为 a[i]。
Byteasar 会按照某个 1 到 n 的全排列 b 走 n-1 次,第 i 次他会从 b[i] 点走到 b[i+1] 点,并且这一次的步伐大小为 c[i]。
对于一次行走,假设起点为 x,终点为 y,步伐为 k,那么 Byteasar 会从 x 开始,每步往前走 k 步,如果最后不足 k 步就能到达 y,那么他会一步走到 y。
请帮助 Byteasar 统计出每一次行走时经过的所有点的权值和。
input
第一行包含一个正整数 n(2<=n<=50000)。表示节点的个数。
第二行包含 n 个正整数,其中第 i 个数为 ai,分别表示每个点的权值。
接下来 n-1 行,每行包含两个正整数 u, v(1<=u,v<=n),表示u与v之间有一条边。
接下来一行包含 n 个互不相同的正整数,其中第i个数为 bi,表示行走路线。
接下来一行包含 n-1 个正整数,其中第 i 个数为 ci,表示每次行走的步伐大小。
output
包含n-1行,每行一个正整数,依次输出每次行走时经过的所有点的权值和
sample input
5
1 2 3 4 5
1 2
2 3
3 4
3 5
4 1 5 2 3
1 3 1 1
sample output
10
6
10
5
@solution@
通过一些简单的树上差分,可以转换为这样一个问题:
若干次询问,第 i 次询问某个点 xi 到根的路径上深度 mod mi = ci 的点的权值和。
一个比较经典的模型。我们将 mi 分两类处理:
(1)\(mi \le \sqrt n\),此时 (mi, ci) 这样一个二元组的数量只有 O(n) 个。我们将这些二元组暴力存下来,每一次加入一个点用 \(O(\sqrt n)\) 的时间去更新。
(2)\(mi > \sqrt n\),此时满足要求的点数量为 \(O(\frac{n}{mi}) = O(\sqrt n)\),暴力将这些点找出来即可。
离线处理,采用 dfs 去统计信息并回答询问。
时间复杂度,第一类为 \(O(n) + O(n\sqrt n)\),第二类为 \(O(n\sqrt n) + O(n)\)。其中前者为询问复杂度,后者为统计信息的复杂度。
总时间复杂度 \(O(n\sqrt n)\)。
@accepted code@
#include<cstdio>
#include<vector>
#include<algorithm>
using namespace std;
const int MAXN = 50000;
const int SQRT = 250;
struct edge{
int to; edge *nxt;
}edges[2*MAXN + 5], *adj[MAXN + 5], *ecnt = &edges[0];
void addedge(int u, int v) {
edge *p = (++ecnt);
p->to = v, p->nxt = adj[u], adj[u] = p;
}
int dep[MAXN + 5], fa[20][MAXN + 5];
void dfs(int rt, int pre) {
dep[rt] = dep[pre] + 1, fa[0][rt] = pre;
for(int i=1;i<20;i++)
fa[i][rt] = fa[i-1][fa[i-1][rt]];
for(edge *p=adj[rt];p;p=p->nxt)
if( p->to != pre ) dfs(p->to, rt);
}
int lca(int u, int v) {
if( dep[u] < dep[v] ) swap(u, v);
for(int i=19;i>=0;i--)
if( dep[fa[i][u]] >= dep[v] )
u = fa[i][u];
if( u == v ) return u;
for(int i=19;i>=0;i--)
if( fa[i][u] != fa[i][v] )
u = fa[i][u], v = fa[i][v];
return fa[0][u];
}
struct query{
int m, r, f;
int num;
query(int _m=0, int _r=0, int _f=0, int _n=0):m(_m), r(_r), f(_f), num(_n){}
};
vector<query>qry[MAXN + 5];
int a[MAXN + 5], b[MAXN + 5], c[MAXN + 5];
int ans[MAXN + 5], arr[MAXN + 5];
int res[SQRT + 5][SQRT + 5];
void dfs2(int x) {
arr[dep[x]] = a[x];
for(int i=1;i<=SQRT;i++)
res[i][dep[x]%i] += a[x];
for(int i=0;i<qry[x].size();i++) {
if( qry[x][i].m <= SQRT ) {
ans[qry[x][i].num] += qry[x][i].f*res[qry[x][i].m][qry[x][i].r];
//printf("%d %d %d\n", qry[x][i].num, x, qry[x][i].f*res[qry[x][i].m][qry[x][i].r]);
}
else {
for(int j=qry[x][i].r;j<=dep[x];j+=qry[x][i].m)
ans[qry[x][i].num] += qry[x][i].f*arr[j];
}
}
for(edge *p=adj[x];p;p=p->nxt)
dfs2(p->to);
for(int i=1;i<=SQRT;i++)
res[i][dep[x]%i] -= a[x];
}
int main() {
int n; scanf("%d", &n);
for(int i=1;i<=n;i++) scanf("%d", &a[i]);
for(int i=1;i<n;i++) {
int u, v; scanf("%d%d", &u, &v);
addedge(u, v);
}
dfs(1, 0);
for(int i=1;i<=n;i++) scanf("%d", &b[i]);
for(int i=1;i<n;i++) scanf("%d", &c[i]);
for(int i=1;i<n;i++) {
int p = b[i], q = b[i+1], r = lca(p, q);
qry[p].push_back(query(c[i], dep[p]%c[i], 1, i));
qry[fa[0][r]].push_back(query(c[i], dep[p]%c[i], -1, i));
int k = (dep[r] + c[i] - (dep[p] - dep[r])%c[i])%c[i];
qry[q].push_back(query(c[i], k, 1, i));
qry[r].push_back(query(c[i], k, -1, i));
if( dep[q]%c[i] != k ) ans[i] += a[q];
}
dfs2(1);
for(int i=1;i<n;i++) printf("%d\n", ans[i]);
}
@details@
看到网上题解说树链剖分什么的,我只想说:
果然离线处理 + 差分好啊,一个 dfs 搞定的事情。
@bzoj - 4381@ [POI2015] Odwiedziny的更多相关文章
- [POI2015]Odwiedziny
[POI2015]Odwiedziny 题目大意: 一棵\(n(n\le5\times10^4)\)个点的树,\(n\)次询问从一个点到另一个点的路径上,每次跳\(k\)个点,所经过的点权和. 思路: ...
- bzoj 4386: [POI2015]Wycieczki
bzoj 4386: [POI2015]Wycieczki 这题什么素质,爆long long就算了,连int128都爆……最后还是用long double卡过的……而且可能是我本身自带大常数吧,T了 ...
- BZOJ 4385: [POI2015]Wilcze doły
4385: [POI2015]Wilcze doły Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 648 Solved: 263[Submit][ ...
- BZOJ 4384: [POI2015]Trzy wieże
4384: [POI2015]Trzy wieże Time Limit: 20 Sec Memory Limit: 128 MBSubmit: 217 Solved: 61[Submit][St ...
- Bzoj 3747: [POI2015]Kinoman 线段树
3747: [POI2015]Kinoman Time Limit: 60 Sec Memory Limit: 128 MBSubmit: 553 Solved: 222[Submit][Stat ...
- BZOJ 3747 POI2015 Kinoman 段树
标题效果:有m点,每个点都有一个权值.现在我们有这个m为点的长度n该序列,寻求区间,它仅出现一次在正确的点区间内值和最大 想了很久,甚至神标题,奔说是水的问题--我醉了 枚举左点 对于每个请求留点右键 ...
- BZOJ 4380 [POI2015]Myjnie | DP
链接 BZOJ 4380 题面 有n家洗车店从左往右排成一排,每家店都有一个正整数价格p[i]. 有m个人要来消费,第i个人会驶过第a[i]个开始一直到第b[i]个洗车店,且会选择这些店中最便宜的一个 ...
- bzoj4381: [POI2015]Odwiedziny
这题搞了我一下午……因为一些傻X的问题…… 对于步长大于sqrt(n)的询问,我们可以直接暴力求解 然后,我们可以事先预处理出d[u][step]表示u往上跳,每次跳step步,直到跳到不能跳为止,所 ...
- BZOJ 3747 POI2015 Kinoman
因为上午没有准备够题目,结果发现写完这道题没题可写了QAQ 又因为这道题范围是100w,我写了发线段树,以为要T,上午就花了一个小时拼命卡常数 结果下午一交居然过了QAQ 我们考虑枚举L,求最大R使得 ...
随机推荐
- jeecms 基本架构研究
最近工作需要内容管理系统,下载了jeecms v5 顺便学习一下它的架构: 采用框架为:Hibernate3.3.2+spring3.05+springMVC+freemarker2.3.16 Hib ...
- maven下载安装以及环境配置
1.到官网下载maven 2.解压到自己想放的安装目录 3.复制maven所在的安装路径,然后右键我的电脑-->属性-->高级系统设置-->环境变量 4.在系统变量中新建,设置变量名 ...
- 【AHOI2013复仇】从一道题来看DFS及其优化的一般步骤和数组分层问题【转】
http://www.cppblog.com/MatoNo1/archive/2012/09/23/191708.html —————————————————————————————————————— ...
- 数据库设计 ch.7
数据库建设的基本规律 三分技术 七分管理 十二分基础数据 阶段 需求分析阶段 概念设计阶段 逻辑设计阶段 物理设计阶段 数据库实施阶段 数据库维护阶段 1 需求分析 2 概念设计 形成概念模型 3 逻 ...
- centos部分网站无法访问问题的解决
CentOS 5内核对TCP的读缓冲区大小有缺省设置,缺省为:net.ipv4.tcp_rmem = 4096 87380 4194304 解决办法就是将最后一个数字改小一点,具体操作就是在文件/et ...
- 设置Linux系统的空闲等待时间TMOUT的方法和Linux反空闲设置的两种方法
为了增强linux系统的安全性,我们需要在用户输入空闲一段时间后自动断开,这个操作可以由设置TMOUT值来实现.将以下字段加入到/etc/profile 中即可(对所有用户生效). export TM ...
- html DOM(CSS放置位置的问题)
转载自: http://www.php.cn/div-tutorial-386900.html (本文对读者有帮助的话请移步支持原作者) 笔记: 这样会先加载css的样式,在渲染dom的时候已经知道了 ...
- laravel-admin 安装(总结)
https://www.jianshu.com/p/844b05e4c45a laravel-admin 是一个可以快速帮你构建后台管理的工具,它提供的页面组件和表单元素等功能,能帮助你使用很少的代码 ...
- 深入剖析Redis RDN持久化机制
rdb是redis保存内存数据到磁盘数据的其中一种方式(另一种是AOF).Rdb的主要原理就是在某个时间点把内存中的所有数据的快照保存一份到磁盘上.在条件达到时通过fork一个子进程把内存中的数据写到 ...
- 2-1 Numpy-数组
(1) 数组的创建 # !usr/bin/env python # Author:@vilicute import numpy as np # 1.用array创建数组并查看数组的属性 arr1 = ...