- 动态开点线段树的做法。虽然通不过1e6的数据,但至少能通过数据规模小一些的原题
点击查看代码
#include <bits/stdc++.h>
using namespace std;
struct t1
{
int l,r,x;
}t[1000000*40+1000000+5];
int tot;
int n,m,q;
int ask(int p,int l,int r,int u,int v,int k)
{
if(!p)
{
return k;
}
if(u<=l&&v>=r)
{
return t[p].x;
}
int mid=(l+r)>>1;
int id=k;
if(v>mid)
{
id=ask(t[p].r,mid+1,r,u,v,id);
}
if(u<=mid)
{
int cur=id,L=1,R=m;
if(id!=k)
{
while(L!=l||R!=mid)
{
int Mid=(L+R)>>1;
if(mid<=Mid)
{
cur=t[cur].l;
R=Mid;
}
else
{
cur=t[cur].r;
L=Mid+1;
}
}
}
else
{
cur=t[p].l;
}
return ask(cur,l,mid,u,v,id);
}
return id;
}
void change(int &p,int l,int r,int x,int k,int va)
{
if(!p)
{
p=++tot;
t[p].l=t[p].r=0;
t[p].x=k;
}
if(l==r)
{
t[p].x=va;
return;
}
int mid=(l+r)>>1;
if(x<=mid)
{
change(t[p].l,l,mid,x,k,va);
}
else
{
change(t[p].r,mid+1,r,x,k,va);
}
int id=k;
if(t[p].r)
{
id=t[t[p].r].x;
}
if(id!=k)
{
int cur=id,L=1,R=m;
while(L!=l||R!=mid)
{
int Mid=(L+R)>>1;
if(mid<=Mid)
{
cur=t[cur].l;
R=Mid;
}
else
{
cur=t[cur].r;
L=Mid+1;
}
if(!cur)
{
break;
}
}
if(cur)
{
id=t[cur].x;
}
}
else if(t[p].l)
{
id=t[t[p].l].x;
}
t[p].x=id;
}
int main()
{
ios::sync_with_stdio(false);
cin.tie(0);
t[0].l=t[0].r=0;
cin>>n>>m>>q;
tot=n;
for(int i=1;i<=n;i++)
{
t[i].l=t[i].r=0;
t[i].x=i;
}
for(int i=1;i<=m;i++)
{
int a,b;
cin>>a>>b;
if(a!=b)
{
change(a,1,m,i,a,b);
change(b,1,m,i,b,a);
}
}
for(int i=1;i<=q;i++)
{
int l,r,k;
cin>>l>>r>>k;
cout<<ask(k,1,m,l,r,k);
cout<<"\n";
}
return 0;
}