#15949 Piet
출처:https://www.acmicpc.net/problem/15949
문제
Piet은 프로그래밍 언어의 하나로, 코드가 N×M의 2차원 이미지로 되어 있는 것이 특징이다. 이름의 유래는 추상화로 유명한 네덜란드의 화가인 피트 몬드리안(Piet Mondrian)이다. 다음의 이미지는 "Hello, world!"를 출력하는 Piet 코드이다.
Piet 프로그램에서 단위 정사각형을 코델(Codel)이라고 하며 각 코델마다 특정 색깔이 칠해져있다. 이는 비트맵 이미지의 픽셀에 대응되는 개념이다. 같은 색깔의 코델이 상하좌우로 연결된 블록이 프로그램의 최소 실행 단위가 된다. 맨 처음 실행되는 블록은 가장 왼쪽 위 코델이 포함된 블록이며, 규칙에 따라 다음에 실행될 블록으로 이동하는 과정을 반복한다.
현재 블록에서 어떤 블록으로 이동할지를 결정하기 위해 DP(Direction Pointer)와 CC(Codel Chooser)라는 두 가지 값이 존재한다. DP의 값은 왼쪽, 오른쪽, 위쪽, 아래쪽 중 하나이고, CC의 값은 왼쪽 혹은 오른쪽 중 하나이다. 프로그램이 처음 실행될 때 DP의 값은 오른쪽, CC의 값은 왼쪽이다. 어떤 블록으로 이동할지를 선택하는 기준은 다음과 같다.
- 현재 블록의 코델들 중에서 DP의 방향으로 가장 멀리 위치한 코델들을 찾는다. 블록이 볼록하지 않은 경우 이 코델들은 연속하지 않을 수 있다.
- 위에서 찾은 코델들 중 DP 방향을 향했을 때 CC의 방향으로 가장 끝에 있는 코델을 선택한다. 이 때 CC의 방향은 상대적인 방향임에 유의하라. 예를 들어, DP가 아래쪽을 가리키고 CC가 오른쪽을 가리키는 경우, 선택되는 코델은 가장 왼쪽에 있는 코델이 된다.
- 위에서 선택한 코델에서 DP의 방향으로 맞닿은 코델이 포함된 블록이 다음 블록이 된다.
이에 대한 예시로, 다음과 같은 그림에서 어떤 블록으로 이동할지를 선택하는 과정을 살펴보자.
각 칸에 적힌 글자는 색깔을 나타내는 값이다. 현재 블록은 A 블록이고, DP의 값(검은 화살표)은 아래쪽, CC의 값(회색 화살표)은 오른쪽인 상태이다. 이 때 A 블록의 코델들 중에서 DP의 방향으로 가장 멀리 위치한 코델은 A 블록에서 가장 아래쪽에 위치한 코델로, 그림에서 점으로 표시된 코델이다. 이 블록들 중 CC의 방향으로 가장 끝에 있는 코델은 그림에서 테두리로 표시된 블록이다. 이 코델의 아래쪽(DP의 방향)으로 맞닿은 코델(빨간색 별표)이 속한 블록인 B 블록이 다음 블록이 된다.
검은색 블록 혹은 이미지의 경계 바깥은 이동할 수 없는 구역이다. 만약 다음으로 이동하려는 블록이 검은색이거나 이미지의 경계를 벗어나는 경우는 다음의 방법으로 진행한다.
- CC의 값이 왼쪽인 경우 오른쪽으로, 오른쪽인 경우 왼쪽으로 바꾼 후 다시 이동을 시도한다.
- CC의 값을 바꾼 후에도 이동할 수 없는 경우, CC의 값을 유지하며 DP를 시계방향으로 회전한 후 다시 이동을 시도한다.
- 시계방향으로 회전한 후에도 이동할 수 없는 경우, 1번으로 되돌아간다.
- 위 과정을 계속 반복하여 총 8가지의 경우를 모두 시도했는데도 이동할 수 있는 블록을 찾지 못한 경우, 프로그램이 종료된다.
예를 들어, 다음과 같은 그림에서 어떤 블록으로 이동할지를 선택하는 과정을 살펴보자.
현재 블록은 A 블록이고, DP의 방향은 위, CC의 방향은 왼쪽인 상태이다. 이번에도 경계에 위치한 코델을 점으로, CC에 의해 선택된 코델을 테두리로 표시하였다. 현재 상태에서 선택된 다음 블록은 검은색 블록이므로 이동할 수 없다. 따라서 CC의 값을 오른쪽으로 바꾼 뒤 다시 다음 블록을 찾는데, 역시 검은색 블록이므로 이동할 수 없다. 이제 DP의 방향을 시계방향으로 회전하여 오른쪽으로 바꾼 뒤 블록을 찾는다(이 때 CC의 값은 바뀌지 않는다). 선택된 코델과 맞닿은 블록은 이미지의 경계 밖이므로 CC의 값을 왼쪽으로 바꾼 뒤 다시 시도한다. 이번에는 다른 코델이 선택되지만 역시 맞닿은 블록이 이미지의 경계 밖이므로 이동할 수 없다. 다시 DP의 방향을 시계방향으로 회전하여 아래로 바꾼 뒤 블록을 찾는다. 이 때 선택된 블록은 검은색 블록으로 역시 이동할 수 없고, CC의 값을 다시 오른쪽으로 바꾸면 B 블록이 선택되고 이는 이동할 수 있는 블록이다. 따라서 다음 블록은 B 블록이 된다. 이상의 과정을 그림으로 나타내면 다음과 같다.
Piet으로 작성된 프로그램이 주어졌을 때, 이 프로그램이 어떤 블록을 거치며 실행되는지 알고 싶어졌다. 입력된 Piet 프로그램에 대해, 프로그램이 종료할 때까지 거치는 블록의 색깔을 출력하는 프로그램을 작성하시오.
* 실제 Piet의 연산 중에는 픽셀의 색깔에 따라 DP 혹은 CC가 바뀌는 연산이 있지만, 이 문제에 한해서는 위에서 설명된 규칙에 의해서만 DP와 CC가 변경되는 것으로 간주한다.
입력
첫 번째 줄에 자연수 N과 M이 주어진다. (1 ≤ N, M ≤ 100)
두 번째 줄부터 N줄에 걸쳐 M개의 글자로 이루어진 문자열이 입력되며, 이는 Piet 프로그램 이미지의 각 코델을 의미한다. 각각의 글자는 알파벳 대문자로 색깔을 나타내는 값이며, X는 검은색을 의미한다. 종료되지 않는 프로그램은 입력되지 않으며, 가장 왼쪽 위 코델은 검은색이 아님이 보장된다.
출력
첫 번째 줄에 프로그램이 종료될 때까지 거치는 블록들의 색깔을 순서대로 출력한다.
풀이
정말 문제에서 요구하는 것을 그대로 구현만 해주면 된다. 요구하는 구현이 정말 많지만 핵심적으로 구현 해야할 것은 앞서 표시한 빨간색 글씨부분만 함수로 잘 구현 해주면 된다.
프로그램이 처음 실행될 때 DP의 값은 오른쪽, CC의 값은 왼쪽이다.
- 현재 블록의 코델들 중에서 DP의 방향으로 가장 멀리 위치한 코델들을 찾는다. 블록이 볼록하지 않은 경우 이 코델들은 연속하지 않을 수 있다.
- 위에서 찾은 코델들 중 DP 방향을 향했을 때 CC의 방향으로 가장 끝에 있는 코델을 선택한다. 이 때 CC의 방향은 상대적인 방향임에 유의하라. 예를 들어, DP가 아래쪽을 가리키고 CC가 오른쪽을 가리키는 경우, 선택되는 코델은 가장 왼쪽에 있는 코델이 된다.
- 위에서 선택한 코델에서 DP의 방향으로 맞닿은 코델이 포함된 블록이 다음 블록이 된다.
검은색 블록 혹은 이미지의 경계 바깥은 이동할 수 없는 구역이다.
- CC의 값이 왼쪽인 경우 오른쪽으로, 오른쪽인 경우 왼쪽으로 바꾼 후 다시 이동을 시도한다.
- CC의 값을 바꾼 후에도 이동할 수 없는 경우, CC의 값을 유지하며 DP를 시계방향으로 회전한 후 다시 이동을 시도한다.
- 시계방향으로 회전한 후에도 이동할 수 없는 경우, 1번으로 되돌아간다.
- 위 과정을 계속 반복하여 총 8가지의 경우를 모두 시도했는데도 이동할 수 있는 블록을 찾지 못한 경우, 프로그램이 종료된다.
코드
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
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
|
// #15949 Piet
// 블록의 색깔 도 고려, 같은 문자(색깔) 인접 해야만 같은 그룹임
#include <iostream>
#include <cstring>
#include <vector>
#include <queue>
using namespace std;
int dir[4][2] = { {0,1}, {1,0}, {0,-1}, {-1,0} };//우하좌상(맨 처음에 dp방향 오른쪽, cp왼)
// 시계방향 회전은 +1, 반시계방향은 ( +3)%4
typedef struct xy {
int x, y;
};
char map[102][102];
int visit[101][101];
int n, m;
int groupcnt=1;// 그룹화를 하기위한 그룹 넘버
vector <char> ans;// 정답을 출력하기 위한 벡터
xy next_move(int fgroupnum,int dp,int cc) {
xy ret;
//같은 그룹 넘버인것 찾아야함
//ㄱ. 현재 블록의 코델들 중에서 DP의 방향으로 가장 멀리 위치한 코델들을 찾는다.
//블록이 볼록하지 않은 경우 이 코델들은 연속하지 않을 수 있다.
int nx, ny;
if (dp == 0) {
if (cc == 1) {
nx = n - 1, ny = m - 1;
}
else if (cc == 3) {
nx = 0, ny = m - 1;
}
}
else if (dp == 1) {
if (cc == 1) {
nx = n - 1, ny = 0;
}
else if (cc == 3) {
nx = n - 1, ny = m - 1;
}
}
else if (dp == 2) {
if (cc == 1) {
nx = 0, ny = 0;
}
else if (cc == 3) {
nx = n - 1, ny = 0;
}
}
else if (dp == 3) {
if (cc == 1) {
nx = 0, ny = m - 1;
}
else if (cc == 3) {
nx = 0, ny = 0;
}
}
int d = (dp + cc) % 4;// 실제 코델을 찾기 위한 방향 값
//ㄴ. 위에서 찾은 코델들 중 DP 방향을 향했을 때 CC의 방향으로 가장 끝에 있는 코델을 선택한다.
//이 때 CC의 방향은 상대적인 방향임에 유의하라.예를 들어, DP가 아래쪽을 가리키고 CC가 오른쪽을 가리키는 경우, 선택되는 코델은 가장 왼쪽에 있는 코델이 된다.
int initx = nx, inity = ny;
while (true) {
if (visit[nx][ny] == fgroupnum)break;
nx -= dir[d][0], ny -= dir[d][1];
if (nx < 0 || ny < 0 || nx >= n || ny >= m) {
initx -= dir[dp][0], inity -= dir[dp][1];
nx = initx, ny = inity;
}
}
//ㄷ. 위에서 선택한 코델에서 DP의 방향으로 맞닿은 코델이 포함된 블록이 다음 블록이 된다.
nx += dir[dp][0], ny += dir[dp][1];
ret = { nx,ny };
return ret;
}
bool check(int nx, int ny) {
if (nx < 0 || ny < 0 || nx >= n || ny >= m || map[nx][ny] == 'X')return false;
else return true;
}
void bfs(int x,int y,char ch) {//그룹화를 위한 함수
queue<xy>q;
visit[x][y] = groupcnt;
q.push({ x,y });
while (!q.empty()) {
int fx = q.front().x;
int fy = q.front().y;
q.pop();
for (int d = 0; d < 4; d++) {
int nx = fx + dir[d][0];
int ny = fy + dir[d][1];
if (nx < 0 || ny < 0 || nx >= n || ny >= m)continue;
if (visit[nx][ny])continue;
if (map[nx][ny] != ch)continue;
visit[nx][ny] = groupcnt;
q.push({ nx,ny });
}
}
}
void simul() {
//맨 처음 실행되는 블록은 가장 왼쪽 위 코델이 포함된 블록
ans.push_back(map[0][0]);
int fgroupnum=visit[0][0];
xy nextpos;
int nx, ny;// 다음 블록의 위치
//DP의 값은 왼쪽, 오른쪽, 위쪽, 아래쪽 중 하나이고, CC의 값은 왼쪽 혹은 오른쪽 중 하나이다.
//프로그램이 처음 실행될 때 DP의 값은 오른쪽, CC의 값은 왼쪽이다.
//어떤 블록으로 이동할지를 선택하는 기준은 다음과 같다.
int dp=0, cc=3;//시계방향으로 회전시킬려면 cc=1이면 되고, 반시계면 3을 더해준 후 %4하면 됨.
while (true) {
nextpos = next_move(fgroupnum, dp, cc);
nx = nextpos.x, ny = nextpos.y;
//cout << "dp " << dp << " cc " << cc << endl;
//cout << "nx " << nx << " ny " << ny <<" char "<<fgroupnum<< endl;
//X는 검은 블록을 의미한다.
//검은색 블록 혹은 이미지의 경계 바깥은 이동할 수 없는 구역이다.만약 다음으로 이동하려는 블록이
//검은색이거나 이미지의 경계를 벗어나는 경우는 다음의 방법으로 진행한다.
if (check(nx, ny) == false) {
//cc의 값을 반대로 바꾸고
// 안되면 dp를 시계방향으로 돌리는 함수 생성
int initdp = dp;
int initcc = cc;
while (true) {
//cout << "in while" << endl;
//1. CC의 값이 왼쪽인 경우 오른쪽으로, 오른쪽인 경우 왼쪽으로 바꾼 후 다시 이동을 시도한다.
if (cc == 1)cc = 3;
else cc = 1;
nextpos = next_move(fgroupnum, dp, cc);
nx = nextpos.x, ny = nextpos.y;
//cout << "dp " << dp << " cc " << cc << " nx " << nx << " ny " << ny << endl;
if (check(nx, ny))break;
//2. CC의 값을 바꾼 후에도 이동할 수 없는 경우,
//CC의 값을 유지하며 DP를 시계방향으로 회전한 후 다시 이동을 시도한다.
dp = (dp + 1) % 4;
nextpos = next_move(fgroupnum, dp, cc);
nx = nextpos.x, ny = nextpos.y;
//cout << "dp " << dp << " cc " << cc << " nx " << nx << " ny " << ny << endl;
if (check(nx, ny))break;
//3. 시계방향으로 회전한 후에도 이동할 수 없는 경우, 1번으로 되돌아간다.
//4. 위 과정을 계속 반복하여 총 8가지의 경우를 모두 시도했는데도
//이동할 수 있는 블록을 찾지 못한 경우, 프로그램이 종료된다.
if (dp == initdp && cc == initcc)return;
}
}
fgroupnum = visit[nx][ny];
ans.push_back(map[nx][ny]);
}
}
int main() {
cin >> n >> m;
for (int i = 0; i < n; i++) {
for (int j = 0; j < m; j++) {
cin >> map[i][j];
}
}
for (int i = 0; i < n; i++) {
for (int j = 0; j < m; j++) {
if (!visit[i][j]) {
bfs(i, j,map[i][j]);
groupcnt++;
}
}
}
simul();
for (int i = 0; i < ans.size(); i++) {
cout << ans[i];
}cout << endl;
}
|
cs |