[BZOJ4012][HNOI2015]开店(动态点分治,树链剖分)
4012: [HNOI2015]开店
Time Limit: 70 Sec Memory Limit: 512 MB
Submit: 2168 Solved: 947
[Submit][Status][Discuss]Description
风见幽香有一个好朋友叫八云紫,她们经常一起看星星看月亮从诗词歌赋谈到
人生哲学。最近她们灵机一动,打算在幻想乡开一家小店来做生意赚点钱。这样的想法当然非常好啦,但是她们也发现她们面临着一个问题,那就是店开在哪里,面向什么样的人群。很神奇的是,幻想乡的地图是一个树形结构,幻想乡一共有 n个地方,编号为 1 到 n,被 n-1 条带权的边连接起来。每个地方都住着一个妖怪,其中第 i 个地方的妖怪年龄是 x_i。妖怪都是些比较喜欢安静的家伙,所以它们并不希望和很多妖怪相邻。所以这个树所有顶点的度数都小于或等于 3。妖怪和人一样,兴趣点随着年龄的变化自然就会变化,比如我们的 18 岁少女幽香和八云紫就比较喜欢可爱的东西。幽香通过研究发现,基本上妖怪的兴趣只跟年龄有关,所以幽香打算选择一个地方 u(u为编号),然后在 u开一家面向年龄在 L到R 之间(即年龄大于等于 L、小于等于 R)的妖怪的店。也有可能 u这个地方离这些妖怪比较远,于是幽香就想要知道所有年龄在 L 到 R 之间的妖怪,到点 u 的距离的和是多少(妖怪到 u 的距离是该妖怪所在地方到 u 的路径上的边的权之和) ,幽香把这个称为这个开店方案的方便值。幽香她们还没有决定要把店开在哪里,八云紫倒是准备了很多方案,于是幽香想要知道,对于每个方案,方便值是多少呢。Input
第一行三个用空格分开的数 n、Q和A,表示树的大小、开店的方案个数和妖
怪的年龄上限。第二行n个用空格分开的数 x_1、x_2、…、x_n,x_i 表示第i 个地点妖怪的年龄,满足0<=x_i<A。(年龄是可以为 0的,例如刚出生的妖怪的年龄为 0。)接下来 n-1 行,每行三个用空格分开的数 a、b、c,表示树上的顶点 a 和 b 之间有一条权为c(1 <= c <= 1000)的边,a和b 是顶点编号。接下来Q行,每行三个用空格分开的数 u、 a、 b。对于这 Q行的每一行,用 a、b、A计算出 L和R,表示询问“在地方 u开店,面向妖怪的年龄区间为[L,R]的方案的方便值是多少”。对于其中第 1 行,L 和 R 的计算方法为:L=min(a%A,b%A),R=max(a%A,b%A)。对于第 2到第 Q行,假设前一行得到的方便值为 ans,那么当前行的 L 和 R 计算方法为: L=min((a+ans)%A,(b+ans)%A),R=max((a+ans)%A,(b+ans)%A)。Output
对于每个方案,输出一行表示方便值。
Sample Input
10 10 10
0 0 7 2 1 4 7 7 7 9
1 2 270
2 3 217
1 4 326
2 5 361
4 6 116
3 7 38
1 8 800
6 9 210
7 10 278
8 9 8
2 8 0
9 3 1
8 0 8
4 2 7
9 7 3
4 7 0
2 2 7
3 2 1
2 3 4Sample Output
1603
957
7161
9466
3232
5223
1879
1669
1282
0HINT
满足 n<=150000,Q<=200000。对于所有数据,满足 A<=10^9
填上两年前的坑。
方法一:
用std::vector记录管辖范围内的点的距离并按年龄排序,因为没有修改所以前缀和加二分求出答案。
想到点分治就要做好近百行代码的觉悟,光RMQLCA+找中心模板就有50行。
ST和lg处理的个数都是tot而不是n!
- #include<cstdio>
- #include<vector>
- #include<algorithm>
- #include<iostream>
- #define rep(i,l,r) for (int i=l; i<=r; i++)
- #define For(i,x) for (int i=h[x],k; i; i=nxt[i])
- #define pb push_back
- typedef long long ll;
- using namespace std;
- const int N=,inf=;
- int n,Q,A,u,v,w,L,R,l,r,cnt,tot,S,rt,vis[N],lg[N<<],pos[N],a[N],h[N],sz[N],f[N],fa[N],to[N<<],val[N<<],nxt[N<<];
- struct D{
- int val; ll dis,sum;
- D(){}; D(int a,int b,int c):val(a),dis(b),sum(c){};
- bool operator <(const D &a)const{ return val==a.val ? dis==a.dis ? sum<a.sum : dis<a.dis : val<a.val; }
- };
- vector<D>va[N],vb[N];
- ll d[N],st[][N<<],ans;
- template<typename T>inline void rd(T &x){
- int t; char ch;
- for (t=; !isdigit(ch=getchar()); t=(ch=='-'));
- for (x=ch-''; isdigit(ch=getchar()); x=x*+ch-'');
- if (t) x=-x;
- }
- void add(int u,int v,int w){ to[++cnt]=v; val[cnt]=w; nxt[cnt]=h[u]; h[u]=cnt; }
- void dfs(int x,int fa){
- st[][++tot]=d[x]; pos[x]=tot;
- For(i,x) if ((k=to[i])!=fa) d[k]=d[x]+val[i],dfs(k,x),st[][++tot]=d[x];
- }
- void rmq(){ rep(i,,) rep(j,,tot-(<<i)+) st[i][j]=min(st[i-][j],st[i-][j+(<<(i-))]); }
- ll dis(int u,int v){
- int x=pos[u],y=pos[v];
- if (x>y) swap(x,y);
- int t=lg[y-x+];
- return d[u]+d[v]-*min(st[t][x],st[t][y-(<<t)+]);
- }
- void find(int x,int fa){
- sz[x]=; f[x]=;
- For(i,x) if ((k=to[i])!=fa && !vis[k]) find(k,x),sz[x]+=sz[k],f[x]=max(f[x],sz[k]);
- f[x]=max(f[x],S-sz[x]);
- if (f[x]<f[rt]) rt=x;
- }
- void solve(int x){
- vis[x]=;
- For(i,x) if (!vis[k=to[i]]) S=sz[k],f[rt=]=inf,find(k,x),fa[rt]=x,solve(rt);
- }
- void work(){
- rep(x,,n) for (int i=x; i; i=fa[i]) va[i].pb(D(a[x],dis(i,x),)),vb[i].pb(D(a[x],dis(x,fa[i]),));
- rep(i,,n){
- va[i].pb(D(-,,)); va[i].pb(D(<<,,));
- vb[i].pb(D(-,,)); vb[i].pb(D(<<,,));
- sort(va[i].begin(),va[i].end()); sort(vb[i].begin(),vb[i].end());
- for (int j=; j<(int)va[i].size(); j++) va[i][j].sum=va[i][j-].sum+va[i][j].dis;
- for (int j=; j<(int)vb[i].size(); j++) vb[i][j].sum=vb[i][j-].sum+vb[i][j].dis;
- }
- }
- ll que(int x,int p){
- ll ans=; p++;
- for (int i=x; i; i=fa[i]){
- int t=lower_bound(va[i].begin(),va[i].end(),D(p,,))-va[i].begin()-;
- ans+=va[i][t].sum+t*dis(i,x);
- }
- for (int i=x; fa[i]; i=fa[i]){
- int t=lower_bound(vb[i].begin(),vb[i].end(),D(p,,))-vb[i].begin()-;
- ans-=vb[i][t].sum+t*dis(fa[i],x);
- }
- return ans;
- }
- int main(){
- rd(n); rd(Q); rd(A);
- lg[]=; rep(i,,n<<) lg[i]=lg[i>>]+;
- rep(i,,n) rd(a[i]);
- rep(i,,n-) rd(u),rd(v),rd(w),add(u,v,w),add(v,u,w);
- dfs(,); rmq(); S=n; f[rt=]=inf; find(,); solve(rt); work();
- while (Q--){
- rd(u); rd(l); rd(r);
- L=min((l+ans)%A,(r+ans)%A); R=max((l+ans)%A,(r+ans)%A);
- printf("%lld\n",ans=que(u,R)-que(u,L-));
- }
- return ;
- }
方法二:
去掉L和R的限制就是裸树剖,用从当前点到根全部打标记询问的时候累计标记的方法求LCA。带上限制直接上主席树就好了。比方法一快三倍。
- #include<cstdio>
- #include<algorithm>
- #define rep(i,l,r) for (int i=l; i<=r; i++)
- #define For(i,x) for (int i=h[x],k; i; i=nxt[i])
- typedef long long ll;
- using namespace std;
- const int N=,M=;
- int n,m,u,v,x,y,w,cnt,mod,k,Q,tot,nd,tim,h[N],nxt[N],to[N],val[N],pos[N],son[N],fa[N],top[N],sz[N],rt[N];
- int ls[M],rs[M],c[M];
- ll sm[M],D[N],d[N],V[N],len[N];
- struct P{ int x,id; }a[N];
- bool cmp(P a,P b){ return a.x<b.x; }
- void add(int u,int v,int w){ to[++cnt]=v; val[cnt]=w; nxt[cnt]=h[u]; h[u]=cnt; }
- int find(int x){
- int l=,r=n+,mid;
- while (l+<r){
- mid=(l+r)>>;
- if (a[mid].x<=x) l=mid; else r=mid;
- }
- return l;
- }
- void dfs(int x,int pre){
- sz[x]=;
- For(i,x) if ((k=to[i])!=pre){
- d[k]=d[x]+val[i]; len[k]=val[i]; fa[k]=x; dfs(k,x); sz[x]+=sz[k];
- if (sz[son[x]]<sz[k]) son[x]=k;
- }
- }
- void dfs2(int x,int tp){
- pos[x]=++tim; V[tim]=len[x]; top[x]=tp;
- if (son[x]) dfs2(son[x],tp);
- For(i,x) if ((k=to[i])!=fa[x] && k!=son[x]) dfs2(k,k);
- }
- void ins(int y,int &x,int L,int R,int l,int r){
- x=++nd; int mid=(L+R)>>;
- sm[x]=sm[y]; c[x]=c[y]; ls[x]=ls[y]; rs[x]=rs[y];
- if (L==l && r==R) { c[x]++; return; }
- sm[x]+=V[r]-V[l-];
- if (r<=mid) ins(ls[y],ls[x],L,mid,l,r);
- else if (l>mid) ins(rs[y],rs[x],mid+,R,l,r);
- else ins(ls[y],ls[x],L,mid,l,mid),ins(rs[y],rs[x],mid+,R,mid+,r);
- }
- ll que(int x,int L,int R,int l,int r){
- ll t=1ll*(V[r]-V[l-])*c[x]; int mid=(L+R)>>;
- if (L==l && r==R) return t+sm[x];
- if (r<=mid) return t+que(ls[x],L,mid,l,r);
- else if (l>mid) return t+que(rs[x],mid+,R,l,r);
- else return t+que(ls[x],L,mid,l,mid)+que(rs[x],mid+,R,mid+,r);
- }
- int main(){
- freopen("bzoj4012.in","r",stdin);
- freopen("bzoj4012.out","w",stdout);
- scanf("%d%d%d",&n,&Q,&mod);
- rep(i,,n) scanf("%d",&a[i].x),a[i].id=i;
- sort(a+,a+n+,cmp);
- rep(i,,n) scanf("%d%d%d",&u,&v,&w),add(u,v,w),add(v,u,w);
- dfs(,); dfs2(,); rep(i,,n) V[i]+=V[i-],D[i]=D[i-]+d[a[i].id];
- rep(i,,n){
- rt[i]=rt[i-];
- for (int x=a[i].id; x; x=fa[top[x]]) ins(rt[i],rt[i],,n,pos[top[x]],pos[x]);
- }
- ll ans=;
- while (Q--){
- scanf("%d",&k); scanf("%d%d",&x,&y); x=(ans+x)%mod; y=(ans+y)%mod;
- if (x>y) swap(x,y);
- x=find(x-); y=find(y); ans=d[k]*(y-x)+D[y]-D[x];
- for (; k; k=fa[top[k]]) ans-=(que(rt[y],,n,pos[top[k]],pos[k])-que(rt[x],,n,pos[top[k]],pos[k]))<<;
- printf("%lld\n",ans);
- }
- return ;
- }
[BZOJ4012][HNOI2015]开店(动态点分治,树链剖分)的更多相关文章
- 洛谷P3676 小清新数据结构题(动态点分治+树链剖分)
传送门 感觉这题做下来心态有点崩……$RMQ$求$LCA$没有树剖快我可以理解为是常数太大……然而我明明用了自以为不会退化的点分然而为什么比会退化的点分跑得反而更慢啊啊啊啊~~~ 先膜一波zsy大佬 ...
- 树分治&树链剖分相关题目讨论
预备知识 树分治,树链剖分 poj1741 •一棵有n个节点的树,节点之间的边有长度.方方方想知道,有多少个点对距离不超过m 题解 点分治模板题.详见我早上写的http://www.cnblogs ...
- [luogu4886] 快递员(点分治,树链剖分,lca)
dwq推的火题啊. 这题应该不算是点分治,但是用的点分治的思想. 每次找重心,算出每一对询问的答案找到答案最大值,考虑移动答案点,使得最大值减小. 由于这些点一定不能在u的两颗不同的子树里,否则你怎么 ...
- 【bzoj4012】[HNOI2015]开店 动态点分治+STL-vector
题目描述 风见幽香有一个好朋友叫八云紫,她们经常一起看星星看月亮从诗词歌赋谈到人生哲学.最近她们灵机一动,打算在幻想乡开一家小店来做生意赚点钱.这样的想法当然非常好啦,但是她们也发现她们面临着一个问题 ...
- P3241 [HNOI2015]开店 动态点分治
\(\color{#0066ff}{ 题目描述 }\) 风见幽香有一个好朋友叫八云紫,她们经常一起看星星看月亮从诗词歌赋谈到人生哲学.最近她们灵机一动,打算在幻想乡开一家小店来做生意赚点钱. 这样的想 ...
- luogu 3241 [HNOI2015]开店 动态点分治+二分+vector
独立写出来+想出来的,1.5h就切了~ 建立点分树,然后用 $vector$ 暴力存所有子节点,然后二分一下子就可以了. #include <cstdio> #include <ve ...
- 【洛谷4719】 动态dp(树链剖分,dp,矩阵乘法)
前言 其实我只是为了过掉模板而写的ddp,实际应用被吊着锤 Solution 并不想写详细的过程 一句话过程:将子树中轻儿子的贡献挂到这个点上面来 详细版:(引用yyb) 总结一下的话,大致的过程是这 ...
- 线段树动态开点+树链剖分BZOJ4999
以每个一个颜色开一颗线段树,内部以dfs序作为线段树节点,权值代表出现次数,维护线段树区间和 #include<iostream> #include<stdio.h> #inc ...
- BZOJ4012[HNOI2015]开店——树链剖分+可持久化线段树/动态点分治+vector
题目描述 风见幽香有一个好朋友叫八云紫,她们经常一起看星星看月亮从诗词歌赋谈到 人生哲学.最近她们灵机一动,打算在幻想乡开一家小店来做生意赚点钱.这样的 想法当然非常好啦,但是她们也发现她们面临着一个 ...
随机推荐
- 第一周 ch01 课下测试
1.Amdahl定律说明,我们对系统的某个部分做出重大改进,可以显著获得一个系统的加速比.(B) A .正确 B .错误 解析:Amdahl定律,该定律的主要思想是,当我们对系统的某个部分加速时,其对 ...
- Let's Encrypt 免费通配 https 签名证书 安装方法
安装环境 centOs7 主要通过 acme.sh (bash脚本)来注册签名 git地址:https://github.com/Neilpang/acme.sh 申请证书流程 1.申请证书-> ...
- 3、CSS基本介绍
1.1 CSS基本介绍一.web 标准所谓的web标准指的就是一系列规范网页书写的要求,它是由W3C组织制定,在它里面要求网页的结构.样式.行为三者相分离.二.名词解释1.结构:就是通过HTML标签搭 ...
- scrapy 爬虫踩过的坑(I)
问题1:正则表达式没问题,但是爬虫进不了item方法 分析: 1. 可能是下载不到list 页面的内容.可以用 scrapy shell url 进行测试 2. 可能是allowed_domains ...
- Ubuntu10.04 下安装RabbitVCS
安装RabbitVCS的方法步骤如下: 1.sudo add-apt-repository ppa:rabbitvcs/ppa #将rabbitvcs的添加到源里面.(次操作会提示是否要添 ...
- sed的额外用法(网摘)
#在我开始动手写一个一个的脚本的时候才会看到更多的用法 1. 在某行的前一行或后一行添加内容(前提是要确定行的内容) # 匹配行前加 sed -i '/allow/ideny' httpd.conf ...
- javadoc生成word接口文档
1.下载DocFlex/Doclet 下载地址 http://www.filigris.com/downloads/ 2.ecplise->project->generate javado ...
- HDU 2053 Switch Game(开灯问题,完全平方数)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=2053 题目大意:灯开始是关着的,有n盏灯,i从1数到n每当灯的序号是i的倍数的时候就对灯进行一次操作( ...
- Fiddler Web Session 列表(1)
Web Session 列表 位置: Web Session 列表 位于Fiddler界面的左侧 ,是Fiddler所抓取到的所有Session会话的列表集合. Web Session 列表 栏名词解 ...
- MySQL之正则表达式
一.介绍 正则表达式用来描述或者匹配符合规则的字符串.它的用法和like比较相似,但是它又比like更强大,能够实现一些很特殊的规则匹配:正则表达式需要使用REGEXP命令,匹配上返回"1& ...