結果
| 問題 |
No.348 カゴメカゴメ
|
| コンテスト | |
| ユーザー |
|
| 提出日時 | 2016-02-27 06:46:03 |
| 言語 | C++11(廃止可能性あり) (gcc 13.3.0) |
| 結果 |
AC
|
| 実行時間 | 197 ms / 2,000 ms |
| コード長 | 3,715 bytes |
| コンパイル時間 | 1,461 ms |
| コンパイル使用メモリ | 108,756 KB |
| 実行使用メモリ | 130,024 KB |
| 最終ジャッジ日時 | 2024-09-24 11:16:54 |
| 合計ジャッジ時間 | 6,433 ms |
|
ジャッジサーバーID (参考情報) |
judge4 / judge2 |
(要ログイン)
| ファイルパターン | 結果 |
|---|---|
| other | AC * 53 |
ソースコード
#include <iostream>
#include <iomanip>
#include <vector>
#include <algorithm>
#include <numeric>
#include <functional>
#include <cmath>
#include <queue>
#include <stack>
#include <set>
#include <map>
#include <sstream>
#include <string>
#define repd(i,a,b) for (int i=(int)(a);i<(int)(b);i++)
#define rep(i,n) repd(i,0,n)
#define all(x) (x).begin(),(x).end()
#define mod 1000000007
#define inf 2000000007
#define mp make_pair
#define pb push_back
typedef long long ll;
using namespace std;
template <typename T>
inline void output(T a, int p) {
if(p) cout << fixed << setprecision(p) << a << "\n";
else cout << a << "\n";
}
// end of template
// union find
class union_find {
public:
int n;
vector<int> parent, rnk, num;
union_find(int n){
parent.resize(n);
rnk.resize(n, 0);
num.resize(n, 1);
rep(i, n){
parent[i] = i;
}
this -> n = n;
}
int root(int x){
if (parent[x] == x) {
return x;
}
else{
return root(parent[x]);
}
}
void unite(int x, int y){
x = root(x);
y = root(y);
if (x == y) {
return;
}
if (rnk[x] < rnk[y]) {
parent[x] = y;
num[y] += num[x];
}
else{
parent[y] = x;
num[x] += num[y];
if (rnk[x] == rnk[y]) {
rnk[x]++;
}
}
n--;
}
bool same(int x, int y){
return root(x) == root(y);
}
int count(){
return n;
}
int count(int x){
return num[root(x)];
}
};
int dx[8] = {1, 0, -1, 0, 1, 1, -1, -1};
int dy[8] = {0, 1, 0, -1, 1, -1, 1, -1};
int H, W;
vector<int> vis;
vector<vector<int>> G, T, dp;
vector<string> S;
inline void dfs(int pos, int ring, union_find &uf){
vis[pos] = 1;
for(int c: G[pos]){
int cx = c % W, cy = c / W;
rep(k, ring ? 8 : 4){
if (cx + dx[k] >= 0 && cx + dx[k] < W && cy + dy[k] >= 0 && cy + dy[k] < H) {
if (S[cy + dy[k]][cx + dx[k]] == ring ? '.' : 'x' && vis[uf.root((cy + dy[k]) * W + cx + dx[k])] == 0) {
T[pos].pb(uf.root((cy + dy[k]) * W + cx + dx[k]));
dfs(uf.root((cy + dy[k]) * W + cx + dx[k]), ring ^ 1, uf);
}
}
}
}
}
inline void dfs_m(int now, union_find &uf){ // 一番浅いところにあるxの輪すべてについてこの操作を行う メモ化再帰
dp[now][1] = uf.count(now);
for(int ch: T[now]) for(int chch : T[ch]){
dfs_m(chch, uf);
dp[now][0] += max(dp[chch][0], dp[chch][1]);
dp[now][1] += dp[chch][0];
}
}
int main() {
cin.tie(0);
ios::sync_with_stdio(0);
// source code
cin >> H >> W;
S.resize(H + 2);
S[0] = S[H + 1] = string(W + 2, '.');
rep(i, H){
cin >> S[i + 1];
S[i + 1] = '.' + S[i + 1] + '.';
}
H += 2, W += 2;
union_find uf(H * W);
rep(y, H) rep(x, W){
rep(k, S[y][x] == 'x' ? 8 : 4){
if (!(x + dx[k] >= 0 && x + dx[k] < W && y + dy[k] >= 0 && y + dy[k] < H)) continue;
if (S[y + dy[k]][x + dx[k]] == S[y][x]){
uf.unite(y * W + x, (y + dy[k]) * W + x + dx[k]);
}
}
}
vis.assign(H * W, 0), G.resize(H * W), T.resize(H * W), dp.resize(H * W, vector<int>(2));
rep(i, H * W) G[uf.root(i)].pb(i);
dfs(uf.root(0), 0, uf);
int ret = 0;
for (int x: T[uf.root(0)]) {
dfs_m(x, uf);
ret += max(dp[x][0], dp[x][1]);
}
output(ret, 0);
return 0;
}