ACM Osenbei(挑戰程序設計競賽)

Osenbei

Time Limit: 3000ms
Memory Limit: 65536KB
This problem will be judged on Aizu. Original ID: 0525
64-bit integer IO format: %lld      Java class name: Main

おせんべい

問題

IOI製菓では,創業以來の伝統の製法で煎餅(せんべい)を焼いている.この伝統の製法は,炭火で一定時間表側を焼き,表側が焼けると裏返して,炭火で一定時間裏側を焼くというものである.この伝統を守りつつ,煎餅を機械で焼いている.この機械は縦 R (1 ≤ R ≤ 10) 行, 橫 C (1 ≤ C ≤ 10000) 列の長方形狀に煎餅を並べて焼く.通常は自動運転で,表側が焼けたら一斉に煎餅を裏返し裏側を焼く.

ある日,煎餅を焼いていると,煎餅を裏返す直前に地震が起こり何枚かの煎餅が裏返ってしまった.幸いなことに炭火の狀態は適切なままであったが,これ以上表側を焼くと創業以來の伝統で定められている焼き時間を超えてしまい,煎餅の表側が焼けすぎて商品として出荷できなくなる.そこで,急いで機械をマニュアル操作に変更し,まだ裏返っていない煎餅だけを裏返そうとした.この機械は,橫の行を何行か同時に裏返したり縦の列を何列か同時に裏返したりすることはできるが,殘念なことに,煎餅を1枚ごと裏返すことはできない.

裏返すのに時間がかかると,地震で裏返らなかった煎餅の表側が焼けすぎて商品として出荷できなくなるので,橫の何行かを同時に1回裏返し,引き続き,縦の何列かを同時に1回裏返して,表側を焼きすぎずに両面を焼くことのできる煎餅,つまり,「出荷できる煎餅」の枚數をなるべく多くすることにした.橫の行を1行も裏返さない,あるいは,縦の列を1列も裏返さない場合も考えることにする.出荷できる煎餅の枚數の最大値を出力するプログラムを書きなさい.

地震の直後に,煎餅が次の図のような狀態になったとする.黒い丸が表側が焼ける狀態を,白い丸が裏側が焼ける狀態を表している.

1行目を裏返すと次の図のような狀態になる.

さらに, 1列目と5列目を裏返すと次の図のような狀態になる.この狀態では,出荷できる煎餅は9枚である.

ヒント

R の上限 10 は C の上限 10000 に比べて小さいことに注意せよ.

入力

入力の1行目には2つの整數 R, C (1 ≤ R ≤ 10, 1 ≤ C ≤ 10 000) が空白を區切りとして書かれている.続く R 行は地震直後の煎餅の狀態を表す. (i+1) 行目 (1 ≤ i ≤ R) には, C 個の整數 ai,1, ai,2, ……, ai,C が空白を區切りとして書かれており, ai,j は i 行 j 列 の煎餅の狀態を表している. ai,j が 1 なら表側が焼けることを, 0 なら裏側が焼けることを表す.

出力

提出する出力ファイルは,出荷できる煎餅の最大枚數だけを含む1行からなる.

入出力の例

入力例1 入力例2
2 5
0 1 0 1 0
1 0 0 0 1  
3 6
1 0 0 0 1 0
1 1 1 0 1 0
1 0 1 1 0 1   
 
出力例1 出力例2
9 
15 

上記問題文と自動審判に使われるデータは、情報オリンピック日本委員會が作成し公開している問題文と採點用テストデータです。

Notes on Submission

標準入出力を行うプログラムを作成して下さい.

上記形式で複數のデータセットが與えられます. C, R がともに 0 のとき入力の終わりを示します.

Sample Input

2 5
0 1 0 1 0
1 0 0 0 1
3 6
1 0 0 0 1 0
1 1 1 0 1 0
1 0 1 1 0 1  
0 0

Sample Output

9
15

Source

題目大意:翻餅子,給你一些餅子數量是W*H,每次可以反一行或一列

問能同時最多使多少個餅子正面朝上,0表示背面,1表示正面


解題思路:深度優先搜索

#include <iostream>
#include <bitset>
#include <cstdio>

using namespace std;

#define MAX_R 10
#define MAX_C 10000

int w,h;
bitset<MAX_C> mat[MAX_R];
int maxcnt;

void DFS2(int cur)
{
	if(cur==w)
	{
		int cnt=0;
		for(int i=0;i<h;i++)
		for(int j=0;j<w;j++)
			if(mat[i][j]) cnt++;

		maxcnt=max(cnt,maxcnt);
		return;
	}

	int cnt0=0,cnt1=0;

	for(int i=0;i<h;i++)
		if(mat[i][cur]) cnt1++;
		else cnt0++;

	if(cnt0>cnt1)
		for(int i=0;i<h;i++) mat[i][cur].flip();

	DFS2(cur+1);
}

void DFS1(int cur)
{
	if(cur==h)
	{
		DFS2(0);
		return;
	}

	mat[cur].flip();
	DFS1(cur+1);
	mat[cur].flip();
	DFS1(cur+1);
}

int main()
{
	scanf("%d%d",&h,&w);

	while(w!=0 || h!=0)
	{
		for(int i=0;i<h;i++)
		{
			mat[i].reset();
			for(int j=0;j<w;j++)
			{
				int val;
				scanf("%d",&val);
				if(val) mat[i][j]=true;
				else mat[i][j]=false;
			}
		}

		maxcnt=0;

		DFS1(0);
		printf("%d\n",maxcnt);

		scanf("%d%d",&h,&w);
	}

	return 0;
}


發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章