AOJ 0525 Osenbei (dfs)

题意:

药药!切克闹! 煎饼果子来一套!有一个烤饼器可以烤r行c列的煎饼,煎饼可以正面朝上(用1表示)也可以背面朝上(用0表示)。一次可将同一行或同一列的煎饼全部翻转。现在需要把尽可能多的煎饼翻成正面朝上,问最多能使多少煎饼正面朝上? 

输入:

多组输入,每组第一行为二整数r, c (1 ≤ r ≤ 10, 1 ≤ c ≤ 10 000),剩下r行c列表示煎饼初始状态。r=c=0表示输入结束。 
输出:

对于每组输入,输出最多能使多少煎饼正面朝上。  (中文参考http://bbs.byr.cn/#!article/ACM_ICPC/73337?au=Milrivel )


由于列的数量极其大 那么dfs搜索行 用row[15]数组标记行是否被翻转 然后枚举列是否需要翻转即可

那么复杂度为O(2^R*R*C) 最坏的情况是 2^10*10*10000 = 10^8左右 - - 可以在1s内通过

还可以使用STL的bitset的flip方法来简化本道题的书写 - -  其实dfs写的好话 代码长度就是一行的差距

AC代码如下:

//
//  AOJ 0525 Osenbei
//
//  Created by TaoSama on 2015-02-21
//  Copyright (c) 2015 TaoSama. All rights reserved.
//
#include <algorithm>
#include <cctype>
#include <cmath>
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <iomanip>
#include <iostream>
#include <map>
#include <queue>
#include <string>
#include <set>
#include <vector>
#define CLR(x,y) memset(x, y, sizeof(x))

using namespace std;
const int INF = 0x3f3f3f3f;
const int MOD = 1e9 + 7;
const int N = 1e5 + 10;

int n, m, ans, row[15];
bool a[15][10005];
void dfs(int r) {
	if(r == n) {
		int ret = 0;
		for(int j = 1; j <= m; ++j) {
			int up = 0;
			for(int i = 1; i <= n; ++i) {
				if(!row[i] && a[i][j] || row[i] && !a[i][j])
					++up;
			}
			ret += max(up, n - up);
		}
		ans = max(ans, ret);
		return;
	}
	row[r] = false;
	dfs(r + 1);
	row[r] = true;
	dfs(r + 1);
}

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

	while(cin >> n >> m && n && m) {
		memset(row, 0, sizeof row);
		for(int i = 1; i <= n; ++i)
			for(int j = 1; j <= m; ++j)
				cin >> a[i][j];
		ans = 0;
		dfs(0);
		cout << ans << endl;
	}
	return 0;
}


  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值