CodeForces - 1016D Vasya And The Matrix

题面在这里!

 

    很明显二进制每一位都是无关的,所以可以先把原问题简化:给矩阵中的每个位置填入0/1,使得特定行/列有奇数个1,其他行/列有偶数个1。

    一个比较好想的方法是对行和列 列出 n+m 个异或方程,其中有 n*m 个变量,随便求出一组解就好了(如果有的话)。

    但这个貌似并不是很好写。。。

    可以把解异或方程转化成 在一个完全二分图(左n个点,右m个点)上选边,每个点可以是黑的(对应行/列要求有奇数个1)或者白的(反之),每选一条边就要把两端的点的黑白性颠倒。

    然后发现这是一个经典问题,显然选的边可以只存在与随便一颗生成树上,因为加入后可以成环的边不会影响答案。

    

    并且可以发现无解当且仅当有奇数个黑点,因为选一条边不会改变图中黑点个数的奇偶性,而我们要求最后所有点都消成白点。

 

    其他情况都有解,这里只从操作意义上解释为什么(但其实方程组的证明也很简单,因为你考虑一下把所有n+m行异或起来,会得到一个全空行,也就是有一个方程其实是没有用的,于是总是有解的)。

    考虑一种构造方案:我们对这颗随便找的生成树做一遍dfs,先把子树中的其他节点都处理成白的之后,如果当前节点是黑的,那么就选它到它father的边;否则不选。

    显然这样做总能保证除了根的其他节点都会变成白色,但又因为选一条边不会改变图中黑点的奇偶性,并且已经保证有偶数个黑点了,所以最后根也一定是白色的。

 

#include<bits/stdc++.h>
#define ll long long
using namespace std;
#define pb push_back
const int N=105;

vector<int> g[N*2];
int n,m,a[N*2],ans[N][N],Xor[N*2],now;

inline void pt(int x,int y){
	if(x>y) swap(x,y);
	ans[x][y-n]|=now;
}

void dfs(int x,int fa){
	for(int i:g[x]) if(i!=fa){
	    dfs(i,x);
	    if(Xor[i]) Xor[x]^=now,pt(x,i);
	}
}

int main(){
	scanf("%d%d",&n,&m),m+=n;
	for(int i=1;i<=m;i++) scanf("%d",a+i);
	
	for(int i=n+1;i<=m;i++) g[1].pb(i),g[i].pb(1);
	for(int i=2;i<=n;i++) g[i].pb(n+1),g[n+1].pb(i);
	
	for(now=1;now<=1e9;now<<=1){
		for(int i=1;i<=m;i++) Xor[i]=a[i]&now;
		dfs(1,0);
		if(Xor[1]){ puts("NO"); return 0;}
	}
	
	puts("YES"),m-=n;
	
	for(int i=1;i<=n;i++){
	    for(int j=1;j<=m;j++) printf("%d ",ans[i][j]);
	    puts("");
	}
	
	return 0;
}

  

posted @ 2018-08-12 17:46  蒟蒻JHY  阅读(223)  评论(0编辑  收藏  举报