junior19

  博客园 :: 首页 :: 博问 :: 闪存 :: 新随笔 :: 联系 :: 订阅 订阅 :: 管理 ::

問題

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
题意:给出n*m的01矩阵,可以反转任意行和列,求最终能得到的最多的1数量。

思路:因为行的范围较小,可以枚举所有行状态,然后在每列中选择最优解,即判断0多还是1多,因此用bitset容器方便处理。

# include <iostream>
# include <cstdio>
# include <bitset>
# include <algorithm>
using namespace std;
int n, m, t, imax;
bitset<10001>b[101];

void dfs(int k)
{
    if(k == n)
    {
        int tmp = 0;
        for(int i=0; i<m; ++i)
        {
            int num = 0;
            for(int j=0; j<n; ++j)
                if(b[j][i]) ++num;
            tmp += max(num, n-num);
        }
        imax = max(imax, tmp);
        return;
    }
    dfs(k+1);
    b[k].flip();
    dfs(k+1);
    b[k].flip();
}

int main()
{
    while(~scanf("%d%d",&n,&m),n+m)
    {
        imax = 0;
        for(int i=0; i<n; ++i)
            for(int j=0; j<m; ++j)
            {
                scanf("%d",&t);
                b[i][j] = t;
            }
        dfs(0);
        printf("%d\n",imax);
    }
    return 0;
}



posted on 2017-04-08 19:59  junior19  阅读(79)  评论(0编辑  收藏  举报