[C++로 풀이] GPS (DP)⭐⭐⭐
Categories: Programmers
Tags: Coding Test Algorithm
📌 GPS
난이도 ⭐⭐⭐
🚀 문제
🚀 내 풀이 ⭕
이 문제는.. 한 1시간 고민하다가 전혀 모르겠어요^_^ 하고는 다른 분들의 풀이를 보고 이해해서 겨우 푼 문제다. DP 문제에 너무 약한 것 같다.. DP 문제라고 감도 못 잡았는데 ㅠ ㅠ.. 큰 일이다. DP 문제 많이 풀어봐야겠다 싶다..!!🔥🔥
#include <vector>
#include <algorithm>
using namespace std;
#define INF 987654321
int solution(int n, int m, vector<vector<int>> edge_list, int k, vector<int> gps_log){
int answer = 0;
// road 는 인접 리스트
// 인덱스는 각 지점에 대응. 지점마다 자신과 인접한 지점들을 추가
// ex) road[2] = [3, 4] 2 지점은 3, 4 지점과 연결된 상태임을 의미함.
vector<vector<int>> road(n + 1);
for (int i = 0; i < edge_list.size(); ++i) {
road[edge_list[i][0]].push_back(edge_list[i][1]);
road[edge_list[i][1]].push_back(edge_list[i][0]);
}
// dp의 모든 원소를 INF 로 초기화 해야함
vector<vector<int>> dp(k, vector<int>(n + 1, INF));
dp[0][gps_log[0]] = 0; // 시작점 고정 (0 초의 시작점의 최소 수정횟수는 0)
for (int t = 1; t < k; ++t) { // t = 0 은 위에서 구했으니 제외 (t = 0 땐 무조건 시작점에서 시작해야 함. 이 경우만 고려 되야 함)
for (int pos = 1; pos <= n; ++pos) {
/* 아래 경우들에서 가장 최소 수정 횟수를 가지는 경우를 그대로 물려 받는다. */
// 1초전에도 pos에 있었던 경우(그대로 머무른 경우)
int minValue = dp[t - 1][pos];
// pos 와 인접한 지점들에서 온 경우
for (int i = 0; i < road[pos].size(); ++i)
minValue = min(dp[t - 1][road[pos][i]], minValue);
if (gps_log[t] != pos)
dp[t][pos] = minValue + 1; // 수정이 필요하다면 +1
else
dp[t][pos] = minValue; // 아니라면 냅둠
}
}
answer = dp[k - 1][gps_log[k - 1]]; // gps_log[k-1] 즉, 마지막 시간대인 k-1 초에 목적지까지 필요한 최소 수정 횟수
if (answer >= INF) answer = -1; // dp[k - 1][gps_log[k - 1]] 가 INF 보다 크다는 것은 수정이 불가능한 경우
return answer;
}
- DP 로 해결하는 하는 이유
- 앞 부분 로그의 오류를 하나 고쳤다면 그 순간부터 경로 자체가 바뀌는 것이나 마찬가지이기 때문에(
edge_list
말고gps_log
의 경로) 그 뒤의 로그들의 수정 여부에도 전부 영향을 끼치게 된다. 앞 로그 중 어느 한 곳을 올바르게 수정하면 뒤의 로그 중에서도 이의 영향으로 수정 안해도 되게끔 된다던지..! 하는 그런 영향이 뒤의 로그들에게 전부 미치게 된다. - 입력 크기가 큰지라 완전탐색을 한다면 시간 초과가 발생할게 분명하다.
t
시간까지 오는데까지pos
지점의 최소 수정 횟수를 구해놓는다면, 그 이후의 시간의 인접 노드 지점의 최소 수정 횟수를 구하는데 쓰이게 된다.- 즉, 부분적인 최적해로 전체적인 최적해를 찾아가는 방식인 것이다.
- 수정할 사항이 없으면 인접한 지점들 중 가장 최소로 수정한 횟수를 물려받으면 되는 것이다.
- 앞 부분 로그의 오류를 하나 고쳤다면 그 순간부터 경로 자체가 바뀌는 것이나 마찬가지이기 때문에(
t
시간엔gps_log[t]
거점에 있었다는 로그 정보가 있으니, 현재edge_list
에서 그 정보대로 따르는게 불가능하다면 수정 횟수를 1 증가시켜 수정해주어야 한다.
dp[t][pos] = t
시간까지(즉, t
번째나 마찬가지) pos
지점으로 오는데까지 최소로 수정한 횟수
(t의 범위는 0~k-1. 문제에선 1~k 로 설명 됐지만 0~k-1로 쓰는게 gps_log
의 인덱스와도 일치하니 편해서 0~k-1로 쓰겠다.)
dp
테이블의 크기는 (k) x (n + 1) 이 된다. (n + 1)인 이유는 거점 이름과 인덱스를 일치시키기 위하여!dp[0][gps_log[0]]
: 0 초까지의 시작 지점으로 오는데까지 최소로 수정한 횟수- dp[0][gps_log[0]] = 0
- 시작 거점과 도착 거점은 고정적이라고 문제에서 제한했다.
- 도착 거점이야 그냥
dp[k-1][gps_log[k-1]]
을 리턴하면 되는데, 시작 거점은 DP 의 시작이니 확실히 고정되어야 한다.
- 도착 거점이야 그냥
- 시작 거점이니 당연히 최소 수정횟수는 0 !
- 시작 거점과 도착 거점은 고정적이라고 문제에서 제한했다.
- dp[0][gps_log[0]] = 0
dp[t][pos]
: t 초까지의(=t번째) pos 지점으로 오는데까지 최소로 수정한 횟수- dp[t][pos] = min(그대로 머물렀을 때의 최소 수정 횟수, 연결된 모든 정점들에서 왔을 때의 최소 수정 횟수) + (수정이 필요한 경우엔 +1)
- 1️⃣ 아래 중 더 작은 값을 가지는 것을 선택하여 대입 받는다. (그 최소 수정 횟수를 그대로 물려받는다.)
dp[t - 1][pos]
- 1초 전에도
pos
에 있었을 때, 즉 그대로 머물렀을 때의 최소 수정 횟수 - 즉, t-1 초까지 pos 지점으로 오는데까지 최소로 수정한 횟수
int minValue = dp[t - 1][pos];
- 1초 전에도
dp[t - 1][road[pos][i]]
road[pos][i]
는pos
지점과 연결된 지점이다.road
인접리스트로 알아낼 수 있다.- for문을 순회하며
pos
와 연결된for (int i = 0; i < road[pos].size(); ++i) minValue = min(dp[t - 1][road[pos][i]], minValue);
- 이렇게 최종적으로
pos
에 그대로 머물렀을 때, 각각pos
와 연결된 지점들에서 왔을 때, 이렇게 모든 후보들 중에서 가장 작은 값의 수정 횟수를 가지는 경우를 그대로 물려 받는다.dp[t][pos] = minValue;
- 2️⃣ 수정이 필요하다면
1
을 더해준다. (1회 수정)- 현재
dp[t][pos]
즉,t
초까지pos
지점으로 오는데까지 최소로 수정한 횟수를 구한 것인데 실제 로그 정보에서t
초에는pos
가 아닌 다른 지점에 있었다면t
초에pos
지점으로 오려면 수정이 필요하다는 뜻이다. - 따라서
gps_log[t]
(로그 상에서t
초에 간 지점)이pos
와 일치하지 않으면1
을 추가로 더해준다. - 수정 필요하지 않으면 이 과정은 그냥 넘어가면 됨.
- DP이다 보니 이렇게 모든 경우를 다 저장해두는 것이다.
if (gps_log[t] != pos) dp[t][pos]++;
- 현재
- 1️⃣ 아래 중 더 작은 값을 가지는 것을 선택하여 대입 받는다. (그 최소 수정 횟수를 그대로 물려받는다.)
- dp[t][pos] = min(그대로 머물렀을 때의 최소 수정 횟수, 연결된 모든 정점들에서 왔을 때의 최소 수정 횟수) + (수정이 필요한 경우엔 +1)
#include <vector>
#include <algorithm>
using namespace std;
#define INF 987654321
int solution(int n, int m, vector<vector<int>> edge_list, int k, vector<int> gps_log){
int answer = 0;
vector<vector<int>> road(n + 1);
for (int i = 0; i < edge_list.size(); ++i) {
road[edge_list[i][0]].push_back(edge_list[i][1]);
road[edge_list[i][1]].push_back(edge_list[i][0]);
}
vector<vector<int>> dp(k + 1, vector<int>(n + 1, INF));
dp[1][gps_log[0]] = 0;
for (int t = 2; t <= k; ++t) {
for (int pos = 1; pos <= n; ++pos) {
int minValue = dp[t - 1][pos];
for (int i = 0; i < road[pos].size(); ++i)
minValue = min(dp[t - 1][road[pos][i]], minValue);
if (gps_log[t - 1] != pos) dp[t][pos] = minValue + 1;
else dp[t][pos] = minValue;
}
}
answer = dp[k][gps_log[k - 1]];
if (answer >= INF) answer = -1;
return answer;
}
이 풀이는 문제에서 주어진대로, t
의 범위를 1 ~ k
로 놓았을 때의 풀이.
🌜 개인 공부 기록용 블로그입니다. 오류나 틀린 부분이 있을 경우
언제든지 댓글 혹은 메일로 지적해주시면 감사하겠습니다! 😄
Leave a comment