StkOvflow

STACK OVERFLOW!

一言(ヒトコト)

这年头 expert 很难么?
——zn

AcWing 1359. 洛谷P1457 城堡

解题思路

这道题目是需要维护各种连通块信息的,所以这里我们可以也用并查集维护。这题我们如果注意一点细节,也是可以让代码变得很简洁的:

1.这道题的输入自带状态压缩,如果一个数a&1=1,那么这个数代表这个格子有西面的墙,东南北也是相似。

2.当移除方法不唯一时,优先选择对应方格区域更靠西的墙
如果仍存在多解,选择对应方格区域更靠南的墙。
在此基础上,还存在多解,那么优先选择北面的墙。
   这就启发着我们对于移除的时候优先从西面枚举,再枚举南面,由于格子都是黏在一起的(某个格子的南墙是它下面那个格子的北墙)所以我们只需要枚举北面和东面(因为条件3对北面有限制)的墙就可以了。

可爱的并查集

初始化不可以把存连通块大小的数组memset成1,因为是按照字节赋值的

代码

#include <iostream>
#include <cstring>

using namespace std;

const int N = 55, M = N * N;
int p[M], n, m, g[N][N], sz[M];

int find(int x) 
{
    if (x == p[x]) return x;
    return p[x] = find(p[x]);
}

int main() 
{
    scanf("%d%d", &m, &n);
    
    for (int i = 0; i < n; i ++ ) 
        for (int j = 0; j < m; j ++ ) 
            scanf("%d", &g[i][j]);
            
    for (int i = 0; i < n * m; i ++ ) p[i] = i, sz[i] = 1;
    
    int cnt = n * m, max_area = 1; 
    int dx[] = {-1, 0}, dy[] = {0, 1}, dw[] = {2, 4};
    
    for (int i = 0; i < n; i ++ ) 
        for (int j = 0; j < m; j ++ ) 
            for (int k = 0; k < 2; k ++ ) 
            {
                if (g[i][j] & dw[k]) continue ;
                int x = i + dx[k], y = j + dy[k];
                if (x < 0 || x >= n || y < 0 || y >= m) continue ;
                
                int u = i * m + j, v = x * m + y;
                u = find(u), v = find(v);
                if (u != v) {
                    p[u] = v, sz[v] += sz[u];
                    -- cnt, max_area = max(max_area, sz[v]);
                }
            }
            
    printf("%d\n%d\n", cnt, max_area);
    
    max_area = 0;
    int rx, ry, rw;
    for (int j = 0; j < m; j ++ ) 
        for (int i = n - 1; ~i; i -- ) 
            for (int k = 0; k < 2; k ++ ) 
            {
                if (!g[i][j] & dw[k]) continue ;
                int x = i + dx[k], y = j + dy[k];
                if (x < 0 || x >= n || y < 0 || y >= m) continue ;
                
                int u = i * m + j, v = x * m + y;
                u = find(u), v = find(v);
                if (u != v) {
                    int ar = sz[v] + sz[u];
                    if (ar > max_area) max_area = ar, rx = i + 1, ry = j + 1, rw = k;
                }
            }
            
    printf("%d\n%d %d %c\n", max_area, rx, ry, (rw ? 'E' : 'N'));
    
    return 0;
}
posted @   StkOvflow  阅读(35)  评论(0编辑  收藏  举报
相关博文:
阅读排行:
· 分享4款.NET开源、免费、实用的商城系统
· 全程不用写代码,我用AI程序员写了一个飞机大战
· MongoDB 8.0这个新功能碉堡了,比商业数据库还牛
· 白话解读 Dapr 1.15:你的「微服务管家」又秀新绝活了
· 上周热点回顾(2.24-3.2)
点击右上角即可分享
微信分享提示