zoukankan      html  css  js  c++  java
  • 【巧妙算法系列】【Uva 11464】

    偶数矩阵(Even Parity, UVa 11464)


    给你一个n×n的01矩阵(每个元素非0即1),你的任务是把尽量少的0变成1,使得每个元素的上、下、左、右的元素(如果存在的话)之和均为偶数。比如,如图1-6(a)所示的矩阵至少要把3个0变成1,最终如图1-6(b)所示,才能保证其为偶数矩阵。



      (a)                 (b)

    【输入格式】

    输入的第一行为数据组数TT≤30)。每组数据的第一行为正整数n(1≤n≤15);接下来的n行每行包含n个非0即1的整数,相邻整数间用一个空格隔开。

    【输出格式】

    对于每组数据,输出被改变的元素的最小个数。如果无解,应输出-1。


    思路:

       这道题是一道经典的枚举+模拟综合运用的算法 

       枚举第一排0 1的变化 以此模拟出第二排至第N排的变化

      复杂度为    o((2^n)*n*n)


    书上的分析:

    也许最容易想到的方法就是枚举每个数字“变”还是“不变”,最后判断整个矩阵是否满足条件。遗憾的是,这样做最多需要枚举2255≈5×1067种情况,实在难以承受。

    注意到n只有15,第一行只有不超过215=32 768种可能,所以第一行的情况是可以枚举的。接下来根据第一行可以完全计算出第二行,根据第二行又能计算出第三行(想一想,如何计算),以此类推,这样,总时间复杂度即可降为O(2n×n2)。



    我的代码如下:

    #include<cstdio>
    #include<cstring>
    using namespace std;
    int map[20][20];
    int tempmap[20][20];
    int ans,n;
    int getans(int KT)
    {
    	memset(tempmap,0,sizeof(tempmap));
    	int cnt=0,temp,k;
    	for(int i=1;i<=n;i++)
    	{
    		temp=KT&1;
    		KT=KT>>1;
    		tempmap[1][i]=temp;
    		if(tempmap[1][i]!=map[1][i]) 
    		  if(temp==1) cnt++;
    		  else return 0;
    	}
    	for(int i=2;i<=n;i++)
    	 for(int j=1;j<=n;j++)
    	  {
    	  k=tempmap[i-1][j-1]+tempmap[i-1][j+1]+tempmap[i-2][j];
    	  if(k%2==1) tempmap[i][j]=1;
    	  else tempmap[i][j]=0;
    	  
    	  if(tempmap[i][j]!=map[i][j])  
    		  if(tempmap[i][j]==1) cnt++;
    		  else return 0;
    	  }
    	if(cnt<ans) ans=cnt;
    	return 0;
    }
    int main()
    {
    	//freopen("a.in","r",stdin);
    	//freopen("a.out","w",stdout);
    	int T;
    	scanf("%d",&T);
    	int maxn;
    	int TT=T;
    	while(T--)
    	{
    		ans=2000000000;
    		scanf("%d",&n);
    		maxn=(1<<n)-1;
    		for(int i=1;i<=n;i++)
    		   for(int j=1;j<=n;j++)
    		   	 scanf("%d",&map[i][j]);
    		for(int i=0;i<=maxn;i++)
    		getans(i);
    		if(ans==2000000000) ans=-1;
    	    printf("Case %d: %d
    ",TT-T,ans);
    	}
    	return 0;
    }



  • 相关阅读:
    C# 泛型约束 xxx<T> Where T:约束(一)
    [cb]UIGrid+UIStretch的自适应
    UIButton vs UIEventListener 事件处理
    NGUI Clip Animation (UI动画)
    TexturePacker的plist切成单独小图片
    Collider Collision 区别
    It’s Time To Think Linq
    [cb]NGUI事件及复杂UI管理
    Unity Mono
    【android】getDimension()、getDimensionPixelOffset()和getDimensionPixelSize()区别详解
  • 原文地址:https://www.cnblogs.com/zy691357966/p/5480463.html
Copyright © 2011-2022 走看看