BZOJ1001/LG4001 「ICPC Beijing2006」狼抓兔子 平面图最小割转对偶图最短路

问题描述

BZOJ1001

LG4001


题解

平面图最小割=对偶图最短路

假设起点和终点间有和其他边都不相交的一条虚边。

如图,平面图的若干条边将一个平面划分为若干个图形,每个图形就是对偶图中的一个点。

对偶图中的每一个点,和它在平面图中每一个相邻的图形间有边,边权为原来分开它们的边的边权。

于是平面图最小割就是对偶图最短路。


\(\mathrm{Code}\)

#include<bits/stdc++.h>
using namespace std;

const int maxn=2*1000*1000+7;
int n,m,S,T;
int Head[maxn],to[maxn*3],Next[maxn*3],tot=1,w[maxn*3];

void addedge(int x,int y,int z){
	to[++tot]=y,Next[tot]=Head[x],Head[x]=tot,w[tot]=z;
}

void add(int x,int y,int z){
	addedge(x,y,z);addedge(y,x,z);
}

void Init(void){
	scanf("%d%d",&n,&m);
}

int id(int x,int y,int type){
	return (x-1)*(m-1)+y+(type-1)*(n-1)*(m-1);
}

void Hori(void){
	for(int i=1,x;i<m;i++){
		scanf("%d",&x);
		add(S,id(1,i,1),x);
	}
	for(int i=2,x;i<n;i++){
		for(int j=1;j<m;j++){
			scanf("%d",&x);
			add(id(i-1,j,2),id(i,j,1),x);
		}
	}
	for(int i=1,x;i<m;i++){
		scanf("%d",&x);
		add(id(n-1,i,2),T,x);
	}
}

void Longi(void){
	for(int i=1,x;i<n;i++){
		scanf("%d",&x);add(T,id(i,1,2),x);
		for(int j=2;j<m;j++){
			scanf("%d",&x);
			add(id(i,j-1,1),id(i,j,2),x);
		}
		scanf("%d",&x);add(id(i,m-1,1),S,x);
	}
}

void Obli(void){
	for(int i=1;i<n;i++){
		for(int j=1,x;j<m;j++){
			scanf("%d",&x);
			add(id(i,j,1),id(i,j,2),x);
		}
	}
}

void Graph_build(void){
	S=(n-1)*(m-1)*2+1,T=S+1;
	Hori();
	Longi();
	Obli();
}

int dis[maxn];
bool vis[maxn];
#define pii(x,y) make_pair(x,y)

void dijkstra(void){
	memset(dis,0x3f,sizeof(dis));
	priority_queue<pair<int,int> >q;
	q.push(pii(0,S));dis[S]=0;
	while(!q.empty()){
		int x=(q.top()).second;q.pop();
		if(vis[x]) continue;vis[x]=1;
		if(x==T) return;
		for(int i=Head[x];i;i=Next[i]){
			int y=to[i];
			if(dis[y]>dis[x]+w[i]){
				dis[y]=dis[x]+w[i];
				q.push(pii(-dis[y],y));
			}
			//if(y==T) return;
		}
	}
}

void One(void){
	int ans=0x3f3f3f3f,x;
	for(int i=1;i<=n;i++) for(int j=1;j<m;j++){
		scanf("%d",&x);ans=min(ans,x);
	}
	for(int i=1;i<n;i++) for(int j=1;j<=m;j++){
		scanf("%d",&x);ans=min(ans,x);
	}
	printf("%d\n",ans);
}

void Work(void){
	if(n==1||m==1){
		One();return;
	}
	Graph_build();
	dijkstra();
	printf("%d\n",dis[T]);
}

int main(){
	Init();
	Work();
	return 0;
}
posted @ 2019-12-14 07:48  览遍千秋  阅读(136)  评论(0编辑  收藏  举报