inflearn logo
강의

강의

N
챌린지

챌린지

멘토링

멘토링

N
클립

클립

로드맵

로드맵

지식공유

10주완성 C++ 코딩테스트 | 알고리즘 코딩테스트

8-P

8-P 문제 질문

해결된 질문

209

이명운

작성한 질문수 28

0

안녕하세요 큰돌선생님 매번 좋은 강의 감사합니다

해당문제를 강의 시청전에 먼저 풀어보았는데 15%에서 오답처리를 받아서 질문드립니다

저는 이동 비용은 + 로 처리하였고, 얻을수 있는 금액은 - 로 처리하였습니다. 거꾸로 처리하였습니다. 또한 선생님처럼 벨만포드 알고리즘을 이용하여 문제를 풀었는데 어느부분이 잘못된것인지 잘 모르겠습니다.

마지막 출력은 bfs를 이용하지 않고 사이클이 있는데 이동가능한 경우(돈 무한), 이동 가능한경우(최대의 돈), 이동할 수 없는 경우(gg) 이렇게 3가지로 나누어 출력하였습니다.

 

http://boj.kr/40828aec1df94d10a50538040db954c3

c++ 코딩-테스트

답변 1

1

큰돌

안녕하세요 명운님 ㅎㅎ

양의 사이클이 있고 도착하거나 도착하지 않는 경우의 수에 대한 로직이 꼬여서 그렇습니다.

    if (q.size() && dist[e] != INF) cout << "Gee" << '\n';
    else {
        if (dist[e] != INF) cout << -dist[e] << '\n';
        else cout << "gg" << '\n';
    }

이로직은 얼핏보면 맞아 보이지만 이경우를 처리하지 못합니다.

양의 사이클은 존재 but, 이 사이클이 도착지점으로까지 연결되지 않을 때도 Gee를 출려할 수 있습니다.

그렇기 때문에 해당 부분을 처리해주어야 합니다. (BFS로)

 

또한

dist의 범위.

즉, n=100이고 100개의 도시 전체가 하나의 큰 사이클을 이룬다고 치면 100*100*1,000,000이 될 수도 있어 int범위를 넘어가게 될 수 있기 때문에 dist는 long long으로 선언해주어야 합니다.

 

제가 좀 다듬어봤는데요 ㅎㅎ

참고부탁드립니다.

#include <bits/stdc++.h>
using namespace std;
typedef long long ll; 
const ll INF = (int)1e18; 
ll V, E, s, e, a, b, c, tmp[104], dist[104], visited[104];
vector<pair<ll, ll>> adj[104];
queue<ll> q;

void bellman() {
    fill(dist, dist + 104, INF);
    dist[s] = tmp[s];
    for (int i = 0; i < V; i++) {
        for (int here = 0; here < V; here++) {
            for (auto there : adj[here]) {
                ll to = there.first;
                ll d = there.second;
                if (dist[here] != INF && dist[to] > dist[here] + d + tmp[to]) {
                    if (i == V - 1) q.push(to);
                    dist[to] = dist[here] + d + tmp[to];
                } 
            }
        }
    }
}

int main() { 
    cin >> V >> s >> e >> E;
    for (int i = 0; i < E; i++) {
        cin >> a >> b >> c;
        adj[a].push_back({b, c});
    }
    for (int i = 0; i < V; i++) {
        int n;
        cin >> n; tmp[i] = -n;
    }
    bellman();
    bool ok = 0; 
    
    while(q.size()){
        int here = q.front(); q.pop();
        for(pair<int, int> there : adj[here]){
            if(there.first == e){
                ok = 1; break;
            }
            if(!visited[there.first]){
                visited[there.first] = 1;
                q.push(there.first);
            }
        }
        if(ok) break;
    }

    if(dist[e] == INF)puts("gg");
    else if(ok)puts("Gee");
    else printf("%lld\n", -dist[e]); 
    return 0;
}

 

또 질문 있으시면 언제든지 질문 부탁드립니다.

좋은 수강평과 별점 5점은 제게 큰 힘이 됩니다. :)

감사합니다.

강사 큰돌 올림.

0

이명운

답변과 추가 코드 정말 감사합니다! 매번 너무 큰 도움이 됩니다ㅎㅎ

4 - A

0

8

1

코딩살구클럽 입장이 안됩니다

0

48

2

4-F 경우의 수 질문입니다.

0

30

2

코딩살구클럽 가입이 안됩니다.

0

63

2

살구 클럽에 대한 질문있습ㄴ디ㅏ

0

52

1

교안 158페이지 문의드립니다

0

43

2

코딩살구클럽 관련 건의사항

0

105

1

코살에 19942 다이어트 문제에 N의 범위가 빠져있슴니다

0

44

1

진행 방법 질문드립니다!

0

78

2

2-I) 왜 이 문제가 그래프이론 카테고리에 있는지 잘 모르겠습니다.

0

63

2

2주차 개념#12 트리 순회

0

32

2

백준사이트가 종료된다고 합니다.

0

307

2

백준 서비스 종료

9

943

1

sk 하이닉스 코테 대비

0

382

2

3-G 최댓값 질문

0

53

1

모듈러 연산 값이 10이 아닌 경우도 있지 않나요?

0

84

2

3-I 코드 질문드립니다.

0

63

2

3-N 질문 있습니다.

0

68

2

학습방법

0

105

2

4-H 질문 있습니다 (코드 리뷰)

0

68

2

코딩테스트 어디까지 준비해야 하는지 질문이 있습니다.

0

179

2

2-O 반례가 무엇일지 어떤 부분이 틀렸는지 잘 모르겠습니다.

0

71

2

2주차 개념 #4-2. 인접행렬 질문있습니다.

0

65

2

1-A 문제풀이 후 궁금한 점이 생겨서 질문드립니다.

0

52

2