蓝桥杯:全球变暖

你有一张某海域NxN像素的照片,"."表示海洋、"#"表示陆地,如下所示:
.......
.##....
.##....
....##.
..####.
...###.
.......
其中"上下左右"四个方向上连在一起的一片陆地组成一座岛屿。例如上图就有2座岛屿。
由于全球变暖导致了海面上升,科学家预测未来几十年,岛屿边缘一个像素的范围会被海水淹没。具体来说如果一块陆地像素与海洋相邻(上下左右四个相邻像素中有海洋),它就会被淹没。
例如上图中的海域未来会变成如下样子:
.......
.......
.......
.......
....#..
.......
.......
请你计算:依照科学家的预测,照片中有多少岛屿会被完全淹没。

【输入格式】
第一行包含一个整数N。 (1 <= N <= 1000)
以下N行N列代表一张海域照片。
照片保证第1行、第1列、第N行、第N列的像素都是海洋。

【输入样例】
7
.......
.##....
.##....
....##.
..####.
...###.
.......

【输出样例】
1

分析:有一种情况是一座岛屿部分淹没后变为两座岛屿,因此计算两遍岛屿数求差的做法是不对的。我的思路是,标记不被淹没的陆地(可以想象为岛屿上的“山”),无山的岛屿即是会被完全淹没的岛屿。
具体做法:标记完成后遍历map,从未被淹没的地块开始dfs岛屿,同时淹没经过的地块,若没有遇到过标记地块,则计数+1。

import java.util.Scanner;

public class Main8 {

    static boolean[][] map, map2;
    
    public static void main(String[] args) {
        //初始化地图,由于内存限制,用bit表示海洋或陆地
        Scanner sc = new Scanner(System.in);
        int n = sc.nextInt();
        sc.nextLine();      
        map = new boolean[n][n];
        for(int i=0; i<n; ++i)
        {
            String s = sc.nextLine();
            for(int j=0; j<n; ++j)
            {
                if(s.charAt(j)=='.')
                    map[i][j]=false;
                else
                    map[i][j]=true;
            }
        }
        sc.close();
        //map2用于标记不被淹没的陆地
        map2 = new boolean[n][n];
        for(int i=0; i<n; ++i)
            for(int j=0; j<n; ++j)
                map2[i][j] = false;
        for(int i=0; i<n; ++i)
            for(int j=0; j<n; ++j)
                if(map[i][j])
                    if(map[i-1][j]&&map[i+1][j]&&map[i][j-1]&&map[i][j+1])
                        map2[i][j]=true;
        
        int cnt=0;
        for(int i=0; i<n; ++i)
            for(int j=0; j<n; ++j)
                if(map[i][j])
                    cnt+=dfs(i,j)?1:0;
        System.out.print(cnt);
    }
    
    public static boolean dfs(int i, int j)
    {
        boolean flag = true;
        map[i][j]=false;
        if(map2[i][j])
            flag = false;
        if(map[i-1][j]) flag &= dfs(i-1, j);
        if(map[i+1][j]) flag &= dfs(i+1, j);
        if(map[i][j-1]) flag &= dfs(i, j-1);
        if(map[i][j+1]) flag &= dfs(i, j+1);
        return flag;
    }

}

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • One 1 the [ðə, ði:] art.这,那 ad.[用于比较级;最高级前] 2 be [bi:,bi]...
    梁培林阅读 9,391评论 0 10
  • 我的童年时光有大风车,有金龟子,也有樱桃小丸子,她给我带来的快乐时光,温暖了我整个童年,小丸子温馨的大家庭,是我最...
    梦里寻花阅读 612评论 0 1
  • 我坐在荷塘的小船里, 你躲在窗口, 望我在离你遥远的地方 写着长长的信, 把信放在小溪里, 信在水波里闪光 每一次...
    张新怡阅读 146评论 0 3
  • 竹阴小径,鲜花簇簇,淡淡清香,棉麻布衣 ,清新小院……久久沉浸梦中,感动于自己的内心深处……。周六,不用上...
    孔素霞阅读 238评论 0 2
  • 今天是今天第一天上班,虚岁27岁的我,还是单身,在自己的计划了,不知道未来会不会遇到自己喜欢的人了,跟家里人说20...
    傻蛋期望温暖阅读 91评论 0 0