bzoj4012: [HNOI2015]开店

传送门:http://www.lydsy.com/JudgeOnline/problem.php?id=4012

思路:首先我们考虑一个简化的问题:

给定一棵树,每次询问所有点到一个点的距离和。


画个图就能知道:距离和=所有点到根的距离和+点数*u到根的距离-每个点与u的lca到根的距离*2


于是问题就成了求lca的dis和


那么我们先对每个点,把它到根的路径覆盖一次,然后询问点u时就是从u向上跳,每次加覆盖次数*边权

用树链剖分即可

但是现在每个点多了一个点权,有了限制条件:只计算点权要在[l,r]内的点到u的距离。

对于区间最大最小等问题我们用的是线段树,对于区间内且权值在[a,b]间的点的询问,我们就用可持久化线段树。


这题也一样,先按点权(年龄)从小到大排序,离散化,按点权顺序把每个点到根的路径覆盖一次。

像对序列的问题一样,询问年龄在[l,r]中的lca的dis和,就是[1,r]的距离和-[1,l]的距离和。

然后就没有然后了


#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
const int maxn=150010,maxm=300010,maxt=10000010;
typedef long long ll;
using namespace std;
int n,m,mod,age[maxn],pre[maxm],now[maxn],son[maxm],val[maxm],tot,tim;
int last[maxn],root[maxn],rt; ll ans,sumdis[maxn],sumE[maxn],dis[maxn];
int top[maxn],dep[maxn],hson[maxn],dfn[maxn],fa[maxn],siz[maxn];
struct Monster{int age,id;}mon[maxn];
void add(int a,int b,int c){pre[++tot]=now[a],now[a]=tot,son[tot]=b,val[tot]=c;}
bool operator<(Monster a,Monster b){return a.age!=b.age?a.age<b.age:a.id<b.id;}

struct TSegment_tree{
	ll sum[maxt];int ls[maxt],rs[maxt],tot,time[maxt];
	int modify(int p,int l,int r,int a,int b){
		int x=++tot,mid=(l+r)>>1;
		ls[x]=ls[p],rs[x]=rs[p],sum[x]=sum[p],time[x]=time[p];
		if (l==a&&r==b){time[x]++;return x;}
		sum[x]+=(sumE[b]-sumE[a-1]);
		if (b<=mid) ls[x]=modify(ls[x],l,mid,a,b);
		else if (a>mid) rs[x]=modify(rs[x],mid+1,r,a,b);
		else ls[x]=modify(ls[x],l,mid,a,mid),rs[x]=modify(rs[x],mid+1,r,mid+1,b);
		return x;
	}
	ll query(int p,int l,int r,int a,int b){
		ll res=1ll*(sumE[b]-sumE[a-1])*time[p];
		if (l==a&&r==b){return res+sum[p];}
		int mid=(l+r)>>1;
		if (b<=mid) return res+query(ls[p],l,mid,a,b);
		else if (a>mid) return res+query(rs[p],mid+1,r,a,b);
		else return res+query(ls[p],l,mid,a,mid)+query(rs[p],mid+1,r,mid+1,b);
	}
}T;

void dfs(int x){
	siz[x]=1;
	for (int y=now[x];y;y=pre[y])if (son[y]!=fa[x]){
		fa[son[y]]=x,dis[son[y]]=dis[x]+val[y],last[son[y]]=val[y];
		dfs(son[y]),siz[x]+=siz[son[y]];
		if (siz[son[y]]>siz[hson[x]]) hson[x]=son[y];
	}
}

void btree(int x,int tp){
	sumE[dfn[x]=++tim]=last[x],top[x]=tp;
	if (hson[x]) btree(hson[x],tp);
	for (int y=now[x];y;y=pre[y])
		if (hson[x]!=son[y]&&son[y]!=fa[x])
			btree(son[y],son[y]);
}

int modify(int x){
	while (top[x]!=1){rt=T.modify(rt,1,n,dfn[top[x]],dfn[x]),x=fa[top[x]];}
	return rt=T.modify(rt,1,n,1,dfn[x]);
}

ll query(int rt,int x){
	ll res=0;
	while (top[x]!=1){res+=T.query(rt,1,n,dfn[top[x]],dfn[x]),x=fa[top[x]];}
	return res+T.query(rt,1,n,1,dfn[x]);
}

int main(){
	scanf("%d%d%d",&n,&m,&mod);
	for (int i=1;i<=n;i++) scanf("%d",&mon[i].age),mon[i].id=i;
	sort(mon+1,mon+1+n);
	for (int i=1,a,b,c;i<n;i++) scanf("%d%d%d",&a,&b,&c),add(a,b,c),add(b,a,c);
	dfs(1),btree(1,1);
	for (int i=1;i<=n;i++) sumE[i]+=sumE[i-1],sumdis[i]=sumdis[i-1]+dis[mon[i].id];
	for (int i=1;i<=n;i++) root[i]=modify(mon[i].id);
	for (int i=1,a,b,u;i<=m;i++){
		scanf("%d%d%d",&u,&a,&b);
		a=(1ll*a+ans)%mod,b=(1ll*b+ans)%mod;
		if (a>b) swap(a,b);
		a=lower_bound(mon+1,mon+1+n,(Monster){a,0})-mon,b=upper_bound(mon+1,mon+1+n,(Monster){b,(int)1e9})-mon-1;
		ans=1ll*(b-a+1)*dis[u]+sumdis[b]-sumdis[a-1]-2ll*((query(root[b],u)-query(root[a-1],u)));
		printf("%lld\n",ans);
	}
	return 0;
}






你可能感兴趣的:(bzoj4012: [HNOI2015]开店)