IOI製菓では,創業以来の伝統の製法で煎餅(せんべい)を焼いている.この伝統の製法は,炭火で必定時間表側を焼き,表側が焼けると裏返して,炭火で必定時間裏側を焼くというものである.この伝統を守りつつ,煎餅を機械で焼いている.この機械は縦 R (1 ≤ R ≤ 10) 行, 横 C (1 ≤ C ≤ 10000) 列の長方形状に煎餅を並べて焼く.一般は自動運転で,表側が焼けたら一斉に煎餅を裏返し裏側を焼く.php
ある日,煎餅を焼いていると,煎餅を裏返す直前に地震が起こり何枚かの煎餅が裏返ってしまった.幸いなことに炭火の状態は適切なままであったが,これ以上表側を焼くと創業以来の伝統で定められている焼き時間を超えてしまい,煎餅の表側が焼けすぎて商品として出荷できなくなる.そこで,急いで機械をマニュアル操做に変更し,まだ裏返っていない煎餅だけを裏返そうとした.この機械は,横の行を何行か同時に裏返したり縦の列を何列か同時に裏返したりすることはできるが,残念なことに,煎餅を1枚ごと裏返すことはできない.ios
裏返すのに時間がかかると,地震で裏返らなかった煎餅の表側が焼けすぎて商品として出荷できなくなるので,横の何行かを同時に1回裏返し,引き続き,縦の何列かを同時に1回裏返して,表側を焼きすぎずに両面を焼くことのできる煎餅,つまり,「出荷できる煎餅」の枚数をなるべく多くすることにした.横の行を1行も裏返さない,あるいは,縦の列を1列も裏返さない場合も考えることにする.出荷できる煎餅の枚数の最大値を出力するプログラムを書きなさい.app
地震の直後に,煎餅が次の図のような状態になったとする.黒い丸が表側が焼ける状態を,白い丸が裏側が焼ける状態を表している.jsp
1行目を裏返すと次の図のような状態になる.spa
さらに, 1列目と5列目を裏返すと次の図のような状態になる.この状態では,出荷できる煎餅は9枚である.code
R の上限 10 は C の上限 10000 に比べて小さいことに注意せよ.orm
入力の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 なら裏側が焼けることを表す.blog
提出する出力ファイルは,出荷できる煎餅の最大枚数だけを含む1行からなる.ip
入力例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 |
上記問題文と自動審判に使われるデータは、情報オリンピック日本委員会が做成し公開している問題文と採点用テストデータです。ci
標準入出力を行うプログラムを做成して下さい.
上記形式で複数のデータセットが与えられます. C, R がともに 0 のとき入力の終わりを示します.
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
9 15
题目大意:翻饼子,给你一些饼子数量是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; }