[JAVA/백준] DFS와 BFS: 단지번호붙이기

👀 문제

https://www.acmicpc.net/problem/2667

👊 도전

1. 설계

  1. DFS를 이용하여 상하좌우 이동할 수 있는 노드로 움직인다.
  2. 주변 노드를 탐색하면서 그 갯수를 세어 ArrayList에 저장한다.
  3. 탐색이 끝나면 ArrayList를 오름차순으로 정렬한 후, 사이즈와 원소 값들을 차례대로 출력한다.

2. 구현 (성공 코드)

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
import java.util.ArrayList;
import java.util.Collections;
import java.util.Scanner;

/**
 * @author HEESOO
 *
 */
public class Main {
	static int n;
	static int cnt;
	static int[][] map;
	static boolean[][] visit;
	public static void dfs(int x, int y) {
		if(visit[x][y]) return;
		
		int[] dotX= {0,1,0,-1};
		int[] dotY= {-1,0,1,0};
		visit[x][y]=true;
		cnt++;
		for(int i=0;i<4;i++) {
			int xx=x+dotX[i];
			int yy=y+dotY[i];
			if(0<=xx&&xx<n&&0<=yy&&yy<n)
				if(map[xx][yy]==1&&!visit[xx][yy])
					dfs(xx, yy);
		}
		
	}
	public static void main(String[] args) {
		// TODO Auto-generated method stub
		Scanner scan = new Scanner(System.in);
		n=scan.nextInt();
		map=new int[n][n];
		visit=new boolean[n][n];
		for(int i=0;i<n;i++) {
			String str=scan.next();
			for(int j=0;j<n;j++) {
				map[i][j]=str.charAt(j)-'0';
			}
		}
		
		ArrayList<Integer> list=new ArrayList<>();
		for(int i=0;i<n;i++) {
			for(int j=0;j<n;j++) {
				if(map[i][j]==1&&!visit[i][j]) {
					cnt=0;
					dfs(i, j);
					list.add(cnt);
				}
			}
		}
		
		Collections.sort(list);
		System.out.println(list.size());
		for(int i=0;i<list.size();i++)
			System.out.println(list.get(i));
	}
}

 

3. 결과

실행결과 🤟 성공 🤟

4. 설명

  1. DFS를 구현한다
    • main의 이중for문에서 방문하지않은 노드를 발견하면 현재 위치 주변 노드들의 갯수를 세기 위해 cnt=0으로 초기화하고, DFS를 호출한다. DFS로 주변 노드 탐색이 끝나면 cnt에 총 갯수가 저장되어 있을 것이고, 그 값을 ArrayList에 저장한다.
    • DFS()에서 dotX, dotY 배열을 선언하여 주변 노드로 움직일 수 있는 총 4가지 방법을 저장한다. 인덱스 0부터 차례대로 상, 우, 하, 좌이다. 이떄 다음 노드로 이동할 좌표 xx, yy가 map범위를 넘어가지는 않는지 체크해야한다.
    • 모든 탐색이 끝나면 list를 오름차순으로 정렬한 후, 그 사이즈와 원소 값들을 차례대로 출력한다.

👏 해결 완료!

참고