其他分享
首页 > 其他分享> > 【线段树优化建图】【P5029 T'ill It's Over】

【线段树优化建图】【P5029 T'ill It's Over】

作者:互联网

【线段树优化建图】【P5029 T'ill It's Over】

P5029 T'ill It's Over

Analysis

每次将一个区间一块进行连边,可以用线段树优化,减少连边数。
具体在线段树每个节点维护一个编号,这个编号用一个tot来分配,在build时,每新建一个节点,tot++。将每个节点与它的左右儿子连边。
需要注意的是,既有将整个区间连向某个点的操作,也有将某个点连向整个区间的操作,因此线段树每个节点,实际需要维护两个编号,分别用来从下向上和从上向下连边。为了方便令这两个编号连续,所以只用维护一个编号。
区间加边时,就将区间在线段树上划分,再根据向上/向下连边即可。

Code

#include<bits/stdc++.h>
using namespace std;
//#define int long long
inline int read()
{
	register int x=0,w=1;
	register char ch=getchar();
	while((ch<'0'||ch>'9')&&ch!='-') ch=getchar();
	if(ch=='-') {w=-1;ch=getchar();}
	while(ch>='0'&&ch<='9') {x=(x<<3)+(x<<1)+(ch^48);ch=getchar();}
	return x*w;
}
inline void write(int x)
{
	if(x<0) putchar('-'),x=~(x-1);
	if(x>9) write(x/10);
	putchar('0'+x%10);
}
const int N=1e6+100,M=9*N,inf=1e7;
int n,m,k,hd[M],cnt=1,now[M],id[N<<2],s,t,tot,d[M];
struct node{
	int nxt,to,w;
}e[M<<1];
void add(int x,int y,int z)
{
	e[++cnt].nxt=hd[x],e[cnt].to=y,e[cnt].w=z,hd[x]=cnt;
	e[++cnt].nxt=hd[y],e[cnt].to=x,e[cnt].w=0,hd[y]=cnt;
}
void build(int x,int l,int r)
{
	id[x]=++tot;tot++;
	if(l==r){
		add(l,id[x],inf);add(id[x]+1,l,inf);return;
	}
	int mid=l+r>>1;
	build(x<<1,l,mid);build(x<<1|1,mid+1,r);
	add(id[x<<1],id[x],inf);add(id[x]+1,id[x<<1]+1,inf);
	add(id[x]+1,id[x<<1|1]+1,inf);add(id[x<<1|1],id[x],inf);
}
void change(int x,int L,int R,int l,int r,int op)
{
	if(L>=l&&R<=r){
		if(op) add(tot,id[x]+1,inf);
		else add(id[x],tot,inf);
		return;
	}
	int mid=L+R>>1;
	if(l<=mid) change(x<<1,L,mid,l,r,op);
	if(r>mid) change(x<<1|1,mid+1,R,l,r,op);
}
queue<int>q;
bool bfs()
{
	while(q.size()) q.pop();
	for(int i=1;i<=tot;++i) d[i]=0,now[i]=hd[i];
	d[s]=1;q.push(s);
	while(q.size())
	{
		int x=q.front();q.pop();
		for(int i=hd[x];i;i=e[i].nxt)
		{
			int y=e[i].to;if(d[y]||e[i].w==0) continue;
			d[y]=d[x]+1;q.push(y);
			if(y==t) return 1;
		}
	}
	return 0;
}
int dinic(int x,int flow)
{
	if(x==t) return flow;
	int rest=flow;
	for(int i=now[x];i&&rest;i=e[i].nxt)
	{
		now[x]=i;int y=e[i].to;
		if(d[y]==d[x]+1){
			int k=dinic(y,min(rest,e[i].w));
			if(k<min(rest,e[i].w)) d[y]=0;
			rest-=k;e[i].w-=k;e[i^1].w+=k;
		}
	}
	return flow-rest;
}
signed main()
{
    n=read();m=read();k=read();
    s=k+1,t=s+1,tot=t+1;
    build(1,1,k);
    for(int i=1;i<=m;++i){
    	int op=read(),l=read(),x,y,u,v;
    	if(op==1){
    		x=y=read(),u=v=read();
		}
		else if(op==2) x=read(),y=read(),u=v=read();
		else if(op==3) x=y=read(),u=read(),v=read();
		else x=read(),y=read(),u=read(),v=read();
		tot++;change(1,1,k,x,y,0);add(tot,tot+1,l);tot++;change(1,1,k,u,v,1);
	}
	add(s,1,n);add(k,t,n);int ans=0;
	while(bfs()) ans+=dinic(s,inf);
	write(ans);
	return 0;
}

标签:连边,ch,int,线段,节点,P5029,建图,编号,ill
来源: https://www.cnblogs.com/glq-Blog/p/16245802.html