Hatena::Grouptopcoder

peryaudoのTopCoder日記

2016-04-16

PalindromeMatrixDiv2

| 12:18

#include <iostream>
#include <string>
#include <vector>
#include <climits>
#include <cassert>

using namespace std;

inline int Inv(int x, int lim) {
  return lim - x - 1;
}

void Print(int x, int lim) {
  for (int i = 0; i < lim; ++i) {
    if (x & (1 << i)) cout<<"1"; else cout<<"0";
  }
  cout<<endl;
}

class PalindromeMatrixDiv2 {
 public:
  int minChange(vector<string> A, int rowCount, int columnCount) {
    int ans = INT_MAX;
    for (int i = 0; i < (1 << A.size()); ++i) {
      if (__builtin_popcount(i) < rowCount) continue;
      for (int j = 0; j < (1 << A[0].size()); ++j) {
        if (__builtin_popcount(j) < columnCount) continue;
        vector<string> ca(A);
        for (int k = 0; k < ca.size(); ++k) {
          for (int l = 0; l < ca[0].size(); ++l) {
            vector<int> maj(2, 0);
            ++maj[ca[k][l] - '0'];
            if (i & (1 << k)) ++maj[ca[k][Inv(l, ca[0].size())] - '0'];
            if (j & (1 << l)) ++maj[ca[Inv(k, ca.size())][l] - '0'];
            if (((j & (1<< l)) && (i & (1 << Inv(k, ca.size())))) ||
                ((i & (1 << k)) && (j & (1 << Inv(l, ca[0].size())))))
              ++maj[ca[Inv(k, ca.size())][Inv(l, ca[0].size())] - '0'];
            char filled = maj[0] < maj[1] ? '1' : '0';
            ca[k][l] = filled;
            if (i & (1 << k)) ca[k][Inv(l, ca[0].size())] = filled;
            if (j & (1 << l)) ca[Inv(k, ca.size())][l] = filled;
            if (((j & (1<< l)) && (i & (1 << Inv(k, ca.size())))) ||
                ((i & (1 << k)) && (j & (1 << Inv(l, ca[0].size())))))
              ca[Inv(k, ca.size())][Inv(l, ca[0].size())] = filled;
          }
        }

        int cur = 0;
        for (int k = 0; k < ca.size(); ++k) {
          for (int l = 0; l < ca[0].size(); ++l) {
            cur += ca[k][l] != A[k][l];
          }
        }
        ans = min(ans, cur);
      }
    }

    return ans;
  }
};

行列が十分に小さいのでpalindromeになる行と列を決め打ちして最小になるものを探す。

コードきたな…

2013-12-14

Single Round Match 600 Division 1

| 04:23

とりあえずEasyがちゃんと通せた(ただしあまりはやくない)し、レーティングもわりと上がったのでよしとしたい。

Mediumさっぱり見当もつかなかった…

Rating: 1221→1282

250: ORSolitaire


#include <cstdio>
#include <cmath>
#include <cstring>
#include <ctime>
#include <iostream>
#include <algorithm>
#include <map>
#include <set>
#include <vector>
#include <sstream>
#include <typeinfo>
#include <climits>
#include <numeric>
#include <cmath>
#include <queue>
#include <stack>

using namespace std;

/*
numbersのうち、使ってもいい部分集合というのを考える
goalで立ってないビットで、要素では立っているというような物が存在しない部分集合がこれに該当する

これらの部分集合の全てのORがgoalにならない場合は、return 0

なる時、goalで立っているあるビットが含まれない部分集合を作るにあたって抜かなければいけない部分集合の中の要素の個数がこたえ

 */
class ORSolitaire {
public:
	int getMinimum(vector<int> numbers, int goal) {
		vector<int> usable;
		for (int i = 0; i < numbers.size(); ++i) {
			bool valid = true;
			for (int j = 0; (1 << j) <= numbers[i] || (1 << j) <= goal; ++j) {
				if (!(goal & (1 << j)) && (numbers[i] & (1 << j))) {
					valid = false;
					break;
				}
			}
			if (valid)
				usable.push_back(numbers[i]);
		}

		int made = 0;
		for (int i = 0; i < usable.size(); ++i) {
			made = made | usable[i];
		}

		if (made != goal) return 0;

		int res = INT_MAX;
		for (int i = 0; (1 << i) <= goal; ++i) {
			if (!(goal & (1 << i))) continue;
			int cur = 0;
			for (int j = 0; j < usable.size(); ++j) {
				if (usable[j] & (1 << i)) {
					++cur;
				}
			}
			res = min(res, cur);
		}

		return res;
	}
};