-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy path미로 탈출.js
232 lines (208 loc) · 6.47 KB
/
미로 탈출.js
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
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
// [문제 이해하기]
// 레버를 당겨 출구를 열고 탈출하는데 걸리는 시간을 출력하는 함수를 구현해라.
//
// 입력: array of strings
// 출력: int, 탈출하는데 걸리는 시간
//
// [시간 복잡도]
// 100 100 이라 고민하지 않아도 될듯
//
// [구체적인 예시]
// console.log(solution(["OOOOL", "OOSOO", "OOOOE"])); => visit 을 공통으로 사용하면 안됨
//
// [문제 세분화]
// function solution(maps) {
// // M. visit 배열 공통으로 쓸거야!
// const [rowLen, colLen] = [maps.length, maps[0].length];
// maps = maps.map((item) => item.split(""));
// const visit = Array.from({ length: rowLen }, () => new Array(colLen).fill(0));
// const dy = [-1, 1, 0, 0];
// const dx = [0, 0, -1, 1];
// // M. 범위 체킹
// const rangeCheck = (y, x) => {
// if (y < 0 || y >= rowLen || x < 0 || x >= colLen) return false;
// if (maps[y][x] === "X") return false;
// return true;
// };
//
// // I. 두개로 분할 S to L, L to E
// // M. 최단 거리는 BFS 로 구현 (queue)
// const BFS = (start, end) => {
// const queue = [[start[0], start[1]]];
//
// while (queue.length) {
// const [i, j] = queue.shift();
//
// if (i === end[0] && j === end[1]) return true;
//
// for (let k = 0; k < 4; k++) {
// const ny = i + dy[k];
// const nx = j + dx[k];
//
// // I. visit 0이 아니면
// if (rangeCheck(ny, nx) && !visit[ny][nx]) {
// visit[ny][nx] = visit[i][j] + 1;
// queue.push([ny, nx]);
// }
// }
// }
//
// return false;
// };
//
// let Sinfo;
// let Linfo;
// let Einfo;
// // I. for 문 돌면서 S, L 를 찾아야지
// for (let i = 0; i < rowLen; i++) {
// for (let j = 0; j < colLen; j++) {
// if (maps[i][j] === "S") Sinfo = [i, j];
// else if (maps[i][j] === "L") Linfo = [i, j];
// else if (maps[i][j] === "E") Einfo = [i, j];
// }
// }
//
// // I. S to L
// const StoL = BFS(Sinfo, Linfo);
// // console.log(visit);
// // I. L to E
// const LtoE = BFS(Linfo, Einfo);
// console.log(visit);
//
// if (!StoL || !LtoE) return -1;
// return visit[Einfo[0]][Einfo[1]];
// }
//
// [문제 세분화]
// function solution(maps) {
// // M. visit 배열 공통으로 쓸거야!
// const [rowLen, colLen] = [maps.length, maps[0].length];
// maps = maps.map((item) => item.split(""));
// const dy = [-1, 1, 0, 0];
// const dx = [0, 0, -1, 1];
// // M. 범위 체킹
// const rangeCheck = (y, x) => {
// if (y < 0 || y >= rowLen || x < 0 || x >= colLen) return false;
// if (maps[y][x] === "X") return false;
// return true;
// };
// // I. 두개로 분할 S to L, L to E
// // M. 최단 거리는 BFS 로 구현 (queue)
// const BFS = (start, end) => {
// const queue = [[start[0], start[1]]];
// const visit = Array.from({ length: rowLen }, () =>
// new Array(colLen).fill(0),
// );
// while (queue.length) {
// const [i, j] = queue.shift();
// if (i === end[0] && j === end[1]) return visit[i][j];
// for (let k = 0; k < 4; k++) {
// const ny = i + dy[k];
// const nx = j + dx[k];
// // I. visit 0이 아니면
// if (rangeCheck(ny, nx) && !visit[ny][nx]) {
// visit[ny][nx] = visit[i][j] + 1;
// queue.push([ny, nx]);
// }
// }
// }
// return -1;
// };
// let Sinfo;
// let Linfo;
// let Einfo;
// // I. for 문 돌면서 S, L 를 찾아야지
// for (let i = 0; i < rowLen; i++) {
// for (let j = 0; j < colLen; j++) {
// if (maps[i][j] === "S") Sinfo = [i, j];
// else if (maps[i][j] === "L") Linfo = [i, j];
// else if (maps[i][j] === "E") Einfo = [i, j];
// }
// }
// // I. S to L
// const StoL = BFS(Sinfo, Linfo);
// // console.log(visit);
// // I. L to E
// const LtoE = BFS(Linfo, Einfo);
// if (StoL === -1 || LtoE === -1) return -1;
// return StoL + LtoE;
// }
// [문제 이해하기]
// 레버를 당겨 미로를 탈출하라. 이때 최대한 빠르게 미로 탈출하는 시간을 반환해라.
// 입력: 2차원 배열
// 출력: 최소 시간, if you can't exit, return -1
// [제약 조건]
// 레버를 열어야 탈출 가능
// 출구는 레버를 당기지 않았더라도 지나갈 수 있음
// [접근법]
// 최소 시간 => BFS
// 레버열기, 레버부터 탈출구로 이동. 총 2번에 걸쳐 BFS를 실행해야함.
// 왜 시간 초과 나는거지? => 방문 표시를 안했기 때문.
// [문제 세분화]
// 1. BFS 구현 => 최단 거리 반환, 각 BFS는 서로 다른 이차원 배열을 사용해야 함.
// 2. 이중 for 문에서 BFS 실행
function solution(maps) {
let answer = 0;
let rowLen = maps.length;
let colLen = maps[0].length;
// 상하좌우
const dx = [0, 0, -1, 1];
const dy = [-1, 1, 0, 0];
const breadthFirstSearch = (x, y, endX, endY) => {
const recordArray = Array.from({ length: rowLen }, () =>
Array(colLen).fill(0)
);
const queue = [[x, y]];
while (queue.length) {
const [x, y] = queue.shift();
// 종료 조건
if (x === endX && y === endY) return recordArray[y][x];
for (let k = 0; k < 4; k++) {
const nx = dx[k] + x;
const ny = dy[k] + y;
// 이거 함수로 빼도 되긴해
if (
nx >= 0 &&
nx < colLen &&
ny >= 0 &&
ny < rowLen &&
maps[ny][nx] !== "X" &&
recordArray[ny][nx] === 0
) {
recordArray[ny][nx] = recordArray[y][x] + 1;
queue.push([nx, ny]);
}
}
}
return -1;
};
// 이중 포문
let startPos;
let leverPos;
let endPos;
for (let y = 0; y < rowLen; y++) {
for (let x = 0; x < colLen; x++) {
if (maps[y][x] === "S") startPos = { x, y };
else if (maps[y][x] === "L") leverPos = { x, y };
else if (maps[y][x] === "E") endPos = { x, y };
}
}
let startToLever = breadthFirstSearch(
startPos.x,
startPos.y,
leverPos.x,
leverPos.y
);
let leverToEnd = breadthFirstSearch(
leverPos.x,
leverPos.y,
endPos.x,
endPos.y
);
if (startToLever === -1 || leverToEnd === -1) return -1;
return startToLever + leverToEnd;
}
// console.log(solution(["OOOOL", "OOSOO", "OOOOE"]));
// console.log(solution(["SOOOL", "XXXXO", "OOOOO", "OXXXX", "OOOOE"]));
// console.log(solution(["LOOXS", "OOOOX", "OOOOO", "OOOOO", "EOOOO"]));
// console.log(solution(["SOOOO", "OOXXX", "OXLOO", "OOOXE"]));