1. 程式人生 > >洛谷5038 [SCOI2012]奇怪的遊戲(二分+網路流+判斷奇偶)

洛谷5038 [SCOI2012]奇怪的遊戲(二分+網路流+判斷奇偶)

寒假的時候就聽過這個題。但是一直沒有寫。
qwq

首先,我們發現題目中的圖是個網格圖,然後每次可以將相鄰兩個格子加一。
很容易就想到是黑白染色。那麼每次操作,就相當於同時操作一個白點,一個黑點。
我們會發現,這樣其實到最終局面的時候,黑點和白點所加的差是相等的,也就是說,我們假設黑點的個數是 n u m 1

num1 ,權值和是 s u m 1 sum1 ,白點的個數是 n u
m 2 num2
,權值和是 s u m 2 sum2
。若最終局面的數字是 x x

n u m 1 × x s u m 1 = n u m 2 × x s u m 2 num1\times x - sum1 =num2 \times x - sum2

那麼 x = s u m 1 s u m 2 n u m 1 n u m 2 x = \frac{sum1-sum2}{num1-num2}

比較容易發現的是,如果 n u m 1 ! = n u m 2 num1!=num2 ,那我們可以直接求出來這個 x x ,然後 c h e c k check 一下就好。

那如果 n u m 1 = = n u m 2 num1==num2 呢?
由於兩個格子的數目是一樣的,所以,只要我們能到達 x x ,我們也一定可以通過一些+的操作,使得局面能到達 x + 1 x+1 。那麼這時候就可以直接二分+ c h e c k check 了。

那麼該如果check呢?
其實也比較簡單。
S , T S\rightarrow白點,黑點\rightarrow T

流量是他們的權值和二分的值的差,然後黑白點之間的流量是 i n f inf ,直接跑 d i n i c dinic 即可。

記得最終的 a n s ans 是操作次數,不是最小的合成的數。

#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cstring>
#include<cmath>
#include<queue>
#include<map>
#include<set>
#define mk make_pair
#define ll long long
#define int long long
using namespace std;
inline int read()
{
  int x=0,f=1;char ch=getchar();
  while (!isdigit(ch)) {if (ch=='-') f=-1;ch=getchar();}
  while (isdigit(ch)) {x=(x<<1)+(x<<3)+ch-'0';ch=getchar();}
  return x*f;
}
const int maxn = 5010;
const int maxm = 2e6+1e2;
const int inf = 2e18;
int point[maxn],nxt[maxm],to[maxm];
int val[maxm],cnt=1,n,m;
int a[100][100];
int col[100][100];
void addedge(int x,int y,int w)
{
	nxt[++cnt]=point[x];
	to[cnt]=y;
	point[x]=cnt;
	val[cnt]=w;
}
void insert(int x,int y,int w)
{
  addedge(x,y,w);
  addedge(y,x,0);
}
int h[maxn];
int s,t;
queue<int> q;
int mx;
int tmp;
bool bfs(int s)
{
	  memset(h,-1,sizeof(h));
	  h[s]=0;
	  q.push(s);
	  while (!q.empty())
	  {
	  	int x=q.front();
	  	q.pop();
	  	for (int i=point[x];i;i=nxt[i])
	  	{
	  		int p  = to[i];
	  		if (h[p]==-1 && val[i]>0)
	  		{
	  			q.push(p);
	  			h[p]=h[x]+1;
			  }
		  }
	  }
	  if (h[t]==-1) return false;
	  return true;
}
int dfs(int x,int low)
{
	if (x==t || low==0) return low;
	int totflow=0;
	for (int i=point[x];i;i=nxt[i])
	{
		int p = to[i];
		if (h[p]==h[x]+1 && val[i]>0)
		{
			int tmp = dfs(p,min(low,val[i]));
			val[i]-=tmp;
			val[i^1]+=tmp;
			totflow+=tmp;
			low-=tmp;
			if(low==0) return totflow;
		}
	}
	if(low>0) h[x]=-1;
	return totflow;
}
int dinic()
{
	int ans=0;
	while (bfs(s))
	{
		ans=ans+dfs(s,inf); 
	}
	return ans;
}
void init()
{
  cnt=1;
  memset(point,0,sizeof(point));
}
int getnum(int x,int y)
{
	return (x-1)*m+y;
}
int dx[5]={0,-1,0,1,0};
int dy[5]={0,0,-1,0,1};
bool check(int mid)
{
	init();
	if (mid<mx) return false;
	int sum1=0,sum2=0;
	for (int i=1;i<=n;i++)
	  for (int j=1;j<=m;j++)
	  {
	  	if (col[i][j]==1) insert(s,getnum(i,j),mid-a[i][j]),sum1+=mid-a[i][j];
	  	else insert(getnum(i,j),t,mid-a[i][j]),sum2+=mid-a[i][j];
	  	//sum+=mid-a[i][j];
	  }
	if (sum1!=sum2) return false;
	for (int i=1;i<=n;i++)
	{
		for (int j=1;j<=m;j++)
		{
			for (int k=1;k<=4;k++)
			{
				int ii = i+dx[k];
				int jj = j+dy[k];
				if(ii<=0 || ii>n || jj<=0 || jj>m) continue;
				if (col[i][j]==2) continue;
				insert(getnum(i,j),getnum(ii,jj),inf);
			}
		}
	}
	tmp=dinic();
	return (tmp==sum1);
}
signed main()
{
  int T=read();
  while (T--)
  {
  	  init();
	  n=read();m=read();
	  mx=0;
	  s=maxn-10;
	  t=s+1;
  	  for (int i=1;i<=n;i++)
  	    for (int j=1;j<=m;j++)
		{
		   if ((i+j)&1)
		     col[i][j]=1;
		   else col[i][j]=2;
		   a[i][j]=read();	
		   mx=max(a[i][j],mx);
		} 
	 int num1=0,num2=0;
	 int sum1=0,sum2=0;
	 for (int i=1;i<=n;i++)
	 {
	 	for (int j=1;j<=m;j++)
	 	{
	 		if (col[i][j]==1)
	 		{
	 			num1++;
	 			sum1+=a[i][j];
			}
			else
			{
				num2++;
				sum2+=a[i][j];
			}
		 }
	 }
	 int ans = -1;
	 int l=0,r=1e18;
	 //cout<<1<<endl;	
	 if (num1==num2)
	 {
	 	while (l<=r)
	 	{
	 		int mid = (l+r) >> 1;
	 		if (check(mid)) ans=tmp,r=mid-1;
	 		else l=mid+1;
	 		//cout<<1<<endl;
		}
	 }
	 else
	 {
	 	int now = (sum1-sum2)/(num1-num2);
	 	if (check(now)) ans=tmp;
	 }
	 cout<<ans<<"\n";
  }
  return 0;
}