結果
| 問題 |
No.348 カゴメカゴメ
|
| コンテスト | |
| ユーザー |
|
| 提出日時 | 2016-02-27 05:36:23 |
| 言語 | C++11(廃止可能性あり) (gcc 13.3.0) |
| 結果 |
WA
|
| 実行時間 | - |
| コード長 | 4,782 bytes |
| コンパイル時間 | 1,735 ms |
| コンパイル使用メモリ | 104,920 KB |
| 実行使用メモリ | 130,080 KB |
| 最終ジャッジ日時 | 2024-09-24 11:16:14 |
| 合計ジャッジ時間 | 6,471 ms |
|
ジャッジサーバーID (参考情報) |
judge4 / judge2 |
(要ログイン)
| ファイルパターン | 結果 |
|---|---|
| other | AC * 20 WA * 33 |
ソースコード
#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;
void dfs(int pos, int ring, union_find &uf){ // ring = 1 : 'x', 0: '.'
vis[pos] = 1;
for(int c: G[pos]){
// cout << pos << ", " << c << endl;
int cx = c % W, cy = c / W;
if (ring == 1) {
rep(k, 8){
if (cx + dx[k] >= 0 && cx + dx[k] < W && cy + dy[k] >= 0 && cy + dy[k] < H) {
// cout << c << ": " << (cy + dy[k]) * W + cx + dx[k] << endl;
if (S[cy + dy[k]][cx + dx[k]] == '.' && vis[uf.parent[(cy + dy[k]) * W + cx + dx[k]]] == 0) {
T[pos].pb(uf.parent[(cy + dy[k]) * W + cx + dx[k]]);
dfs(uf.parent[(cy + dy[k]) * W + cx + dx[k]], 0, uf);
}
}
}
}
else{
rep(k, 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]] == 'x' && vis[uf.parent[(cy + dy[k]) * W + cx + dx[k]]] == 0) {
T[pos].pb(uf.parent[(cy + dy[k]) * W + cx + dx[k]]);
dfs(uf.parent[(cy + dy[k]) * W + cx + dx[k]], 1, uf);
}
}
}
}
}
}
void dfs_m(int now, union_find &uf){ // 一番浅いところにあるxの輪すべてについてこの操作を行う メモ化再帰
dp[now][1] = uf.num[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){
if (S[y][x] == 'x') {
rep(k, 8){
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]] == 'x'){
uf.unite(y * W + x, (y + dy[k]) * W + x + dx[k]);
}
}
}
else{ // '.'
rep(k, 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]] == '.'){
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.parent[i]].pb(i);
dfs(uf.parent[0], 0, uf);
int ret = 0;
for (int x: T[uf.parent[0]]) {
dfs_m(x, uf);
ret += max(dp[x][0], dp[x][1]);
}
output(ret, 0);
return 0;
}