我是靠谱客的博主 细腻纸飞机,最近开发中收集的这篇文章主要介绍2019.03.09 bzoj5371: [Pkusc2018]星际穿越(主席树),觉得挺不错的,现在分享给大家,希望可以做个参考。

概述

传送门
题意简述:
给一个序列,对于第iii个位置,它跟[limi,i−1][lim_i,i-1][limi,i1]这些位置都存在一条长度为111的无向边。
dist(u,v)dist(u,v)dist(u,v)表示(u,v)(u,v)(u,v)间最短路长度。
qqq次询问,每次给出l,r,xl,r,xl,r,x,求∑i=lrdist(i,x)sum_{i=l}^rdist(i,x)i=lrdist(i,x)


思路:
有一个显然的结论,从iii走到它之前的点,要么向右边走一次之后一直向左走,要么一直向左走。
对每个点dpdpdp出它最多向右边走一次然后向左走一步能够到达的最小编号,称为mnimn_imni
这样对于编号在[limi,i−1][lim_i,i-1][limi,i1]间的点只用直接走一次,对于编号在[1,limi−1][1,lim_i-1][1,limi1]的点可以先走到mnimn_imni,然后走到iii,这样用主席树维护[1,i−1][1,i-1][1,i1]iii的最短距离,iii对应的线段树用mnimn_imni的更新过来即可。
提示:bzoj轻微卡主席树,最好用fread优化
代码:

#include<bits/stdc++.h>
#define ri register int
using namespace std;
const int rlen=1<<18|1;
inline char gc(){
	static char buf[rlen],*ib,*ob;
	(ib==ob)&&(ob=(ib=buf)+fread(buf,1,rlen,stdin));
	return ib==ob?-1:*ib++;
}
inline int read(){
	int ans=0;
	char ch=gc();
	while(!isdigit(ch))ch=gc();
	while(isdigit(ch))ans=((ans<<2)+ans<<1)+(ch^48),ch=gc();
	return ans;
}
const int N=3e5+5;
int rt[N],lim[N],n,mn[N];
inline int gcd(int a,int b){while(b){int t=a;a=b,b=t-t/a*a;}return a;}
namespace SGT{
	#define lc (son[p][0])
	#define rc (son[p][1])
	#define mid (l+r>>1)
	int sum[N*30],add[N*30],son[N*30][2],tot=0;
	inline void update(int&p,int l,int r,int ql,int qr){
		int o=++tot;
		sum[o]=sum[p],add[o]=add[p],son[o][0]=lc,son[o][1]=rc;
		p=o;
		if(ql<=l&&r<=qr){++add[p];return;}
		sum[p]+=min(qr,r)-max(ql,l)+1;
		if(qr<=mid)update(lc,l,mid,ql,qr);
		else if(ql>mid)update(rc,mid+1,r,ql,qr);
		else update(lc,l,mid,ql,mid),update(rc,mid+1,r,mid+1,qr);
	}
	inline int query(int p,int l,int r,int ql,int qr){
		if(!p)return 0;
		int ret=add[p]*(min(qr,r)-max(ql,l)+1);
		if(ql<=l&&r<=qr)return sum[p]+ret;
		if(qr<=mid)return query(lc,l,mid,ql,qr)+ret;
		if(ql>mid)return query(rc,mid+1,r,ql,qr)+ret;
		return query(lc,l,mid,ql,mid)+query(rc,mid+1,r,mid+1,qr)+ret;
	}
	#undef lc
	#undef rc
	#undef mid
}
int main(){
	n=read();
	for(ri i=2;i<=n;++i)lim[i]=mn[i]=read();
	for(ri i=n-1;i;--i)mn[i]=min(mn[i+1],mn[i]);
	for(ri i=2;i<=n;++i)rt[i]=rt[mn[i]],SGT::update(rt[i],1,n,1,i-1);
	for(ri tt=read(),ans,len,g,l,r,x;tt;--tt){
		l=read(),r=read(),x=read();
		ans=len=r-l+1;
		if(l<lim[x])ans+=SGT::query(rt[lim[x]],1,n,l,min(lim[x]-1,r));
		g=gcd(ans,len);
		cout<<ans/g<<'/'<<len/g<<'n';
	}
	return 0;
}

转载于:https://www.cnblogs.com/ldxcaicai/p/10582378.html

最后

以上就是细腻纸飞机为你收集整理的2019.03.09 bzoj5371: [Pkusc2018]星际穿越(主席树)的全部内容,希望文章能够帮你解决2019.03.09 bzoj5371: [Pkusc2018]星际穿越(主席树)所遇到的程序开发问题。

如果觉得靠谱客网站的内容还不错,欢迎将靠谱客网站推荐给程序员好友。

本图文内容来源于网友提供,作为学习参考使用,或来自网络收集整理,版权属于原作者所有。
点赞(57)

评论列表共有 0 条评论

立即
投稿
返回
顶部