Osenbei 翻馅饼



Description

問題

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 50 1 0 1 01 0 0 0 13 61 0 0 0 1 01 1 1 0 1 01 0 1 1 0 1
 
出力例1出力例2
915

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

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 

题目大意, 看样例1
有一个 2*5的矩阵,然后里面0代表反面,1代表证明,你可以进行如下操作:
任意选取一行(列),将这一行(列)翻转(这一行(列)所有 正面变反面,反面变正面 )
然后可以进行无限次翻转, 要求得是 进行完这些翻转后,得到整个矩形 要求正面的数量最大,求出这个矩形最多的正面

行是1-10 列数是1-10000 

思路:
二维的题,根据经验,都是先压缩成一维的再处理,
这道题也比较类似,因为行最多才10
所以用dfs可以枚举出行翻转的状态,在行已经完全翻转的情况下,枚举列得到每一列翻转和不翻转的最大值
行翻转状态可由一个bool 数组row来记录,如果这一行翻转了, 那么这一行的每个元素1(true) 0(false)
当!row[i] == num[i][j] 就表示正面(因为是枚举 所以 row[i] == num[i][j]也同样成立)
代码如下:
 //
// Create by 神舟 on 2015-02-23
//

#include <iostream>
#include <cstdio>
#include <cstring>
#include <cmath>
#include <cstdlib>
#include <algorithm>
#include <cctype>
#include <stack>
#include <queue>
#include <map>
#include <string>
#include <set>
#include <vector>
using namespace std;

#define CLR(x) memset(x,0,sizeof x)
#define ll long long
const int inf=0x3f3f3f3f;
const int maxn=1e4+5;
const int MOD=5e5+5;
bool mp[12][maxn];
bool row[15];//表示枚举的行是否翻转 true表示翻转
int ans,n,m;

void dfs(int step)
{
    if(step == n){
        int cnt = 0;
        for(int j = 1; j <= m; j++){
            int tmp = 0;
            for(int i = 1; i <= n; i++){
                if(!row[i] == mp[i][j])tmp++;
            }
            cnt += max(tmp, n - tmp);
        }
        ans = max(ans, cnt);
        return;
    }
    
    dfs(step + 1);//枚举不翻转的状态
    row[step] = !row[step];
    dfs(step + 1);//枚举翻转的状态
    row[step] = !row[step];//转回来
}

int main()
{
#ifdef LOCAL
 freopen("in.txt","r",stdin);
 //freopen("out.txt","w",stdout);
#endif
 ios_base::sync_with_stdio(0);


 while(scanf("%d%d",&n,&m) != EOF && n && m){
        CLR(mp);
        CLR(row);
        ans=0;
        for(int i = 1; i <= n; i++) for(int j = 1; j <= m; j++){
            int x;scanf("%d", &x);
            if(x) mp[i][j] = true;
        }
        dfs(0);
        printf("%d\n", ans);
 }
 return 0;
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值