题目大意是讲一个烧饼铺,在一个n * m (1<=n<=10,1<=m<=10000)的烤桌上面摆着一堆烧饼,数字1表示烧饼正面,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比较小,所以可以对行DFS,模拟对行的所有操作。
那列呢?其实列很好处理,对每一列统计1的个数或者0的个数,保留最大者即是最大的正面个数: 试想如果当前列正面个数多,那这一列就不翻面; 如果反面多,那么将该列翻面即可使得原先反面变成正面,所以对列直接统计即可。
这题需要注意的是无论哪一行或者哪一列先翻面不影响结果,即翻面的顺序不影响结果,只考虑该行或该列是否要翻面即可,所以可以直接DFS。
AC代码
#include<iostream>
#include<algorithm>
using namespace std;
int n, m;
char map[11][10001];
int ans;//保存结果
void change(int a) {//翻面函数
for (int i = 0; i < m; i++) {
if (map[a][i] == '1') { map[a][i] = '0'; }
else {
map[a][i] = '1';
}
}
}
void DFS(int a) {//a代表第几行(0 ~ n-1)
if (a == n) {//列举到最后再统计
int tmp = 0;
for (int j = 0; j < m; j++) {
int tmple = 0;
for (int i = 0; i < n; i++) {
if (map[i][j] == '1') tmple++;
}
tmp += max(tmple, n - tmple);
}
ans = max(ans, tmp);
return;
}
DFS(a + 1);
change(a);
DFS(a + 1);
return;
}
int main() {
while (cin >> n >> m) {
if (n == 0 && m == 0) { break; }
for (int i = 0; i < n; i++) {
for (int j = 0; j < m; j++) {
cin >> map[i][j];
}
}
ans = 0;
DFS(0);
cout << ans << endl;
}
}