清新友好的题目
跑一个最短路,然后对海拔建Kruskal重构树,从最后接上去的边(最低的一个)开始DFS一下处理子树里路程的最小值。
询问是每次在重构树上倍增找到深度最浅的海拔高于当天水位线的节点,其子树内的点必定可以通过乘车互相到达。
#include<queue>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
const int N=,M=,X=,inf=2e9;
struct a
{
int x,y,s;
}mst[N];
struct b{int node,dist;};
bool operator < (b x,b y)
{
return x.dist>y.dist;
}
priority_queue<b> hp;
int p[N],noww[M],goal[M],val[M];
int P[N],Noww[M],Goal[M],mul[N][X];
int vis[N],dis[N],aset[N],elev[N],mind[N];
int n,m,T,Q,S,K,t1,t2,t3,t4,cnt,Cnt,tot,ans;
bool cmp(a x,a y)
{
return x.s>y.s;
}
int Finda(int x)
{
return x==aset[x]?x:aset[x]=Finda(aset[x]);
}
void Link(int f,int t,int v)
{
noww[++cnt]=p[f],p[f]=cnt;
goal[cnt]=t,val[cnt]=v;
noww[++cnt]=p[t],p[t]=cnt;
goal[cnt]=f,val[cnt]=v;
}
void Linka(int f,int t)
{
Noww[++Cnt]=P[f];
Goal[Cnt]=t,P[f]=Cnt;
Noww[++Cnt]=P[t];
Goal[Cnt]=f,P[t]=Cnt;
}
void Init()
{
memset(p,,sizeof p);
memset(P,,sizeof P);
memset(mul,,sizeof mul);
memset(vis,,sizeof vis);
memset(dis,0x3f,sizeof dis);
for(int i=;i<=n;i++) aset[i]=i;
cnt=Cnt=ans=,tot=n,dis[]=,hp.push((b){,});
}
void Dijkstra()
{
while(!hp.empty())
{
b tt=hp.top(); hp.pop(); int tn=tt.node;
if(vis[tn]) continue; vis[tn]=true;
for(int i=p[tn];i;i=noww[i])
if(dis[goal[i]]>dis[tn]+val[i])
{
dis[goal[i]]=dis[tn]+val[i];
hp.push((b){goal[i],dis[goal[i]]});
}
}
}
void DFS(int nde,int fth)
{
mul[nde][]=fth;
for(int i=;mul[nde][i-];i++)
mul[nde][i]=mul[mul[nde][i-]][i-];
mind[nde]=(nde<=n)?dis[nde]:inf;
for(int i=P[nde];i;i=Noww[i])
if(Goal[i]!=fth)
{
DFS(Goal[i],nde);
if(mind[Goal[i]]<mind[nde])
mind[nde]=mind[Goal[i]];
}
}
int main()
{
scanf("%d",&T);
while(T--)
{
scanf("%d%d",&n,&m),Init();
for(int i=;i<=m;i++)
{
scanf("%d%d%d%d",&t1,&t2,&t3,&t4);
mst[i]=(a){t1,t2,t4},Link(t1,t2,t3);
}
Dijkstra(),sort(mst+,mst++m,cmp);
for(int i=;i<=m;i++)
{
int tx=Finda(mst[i].x),ty=Finda(mst[i].y),ts=mst[i].s;
if(tx!=ty)
{
elev[++tot]=ts,aset[tot]=aset[tx]=aset[ty]=tot;
Linka(tx,tot),Linka(ty,tot);
}
}
DFS(tot,),scanf("%d%d%d",&Q,&K,&S);
while(Q--)
{
scanf("%d%d",&t1,&t2);
t1=(t1+K*ans-)%n+,t2=(t2+K*ans)%(S+);
for(int i=;~i;i--)
if(elev[mul[t1][i]]>t2) t1=mul[t1][i];
printf("%d\n",ans=mind[t1]);
}
}
return ;
}