Range Swap Query

  • 动态开点线段树的做法。虽然通不过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;
}
posted @ 2024-11-06 11:23  D06  阅读(3)  评论(0编辑  收藏  举报