ABOUT ME

-

Today
-
Yesterday
-
Total
-
  • [BOJ 11501, C++, Python 3] 주식
    알고리즘/BOJ 2021. 8. 3. 15:44
    반응형

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

     

    11501번: 주식

    입력의 첫 줄에는 테스트케이스 수를 나타내는 자연수 T가 주어진다. 각 테스트케이스 별로 첫 줄에는 날의 수를 나타내는 자연수 N(2 ≤ N ≤ 1,000,000)이 주어지고, 둘째 줄에는 날 별 주가를 나타

    www.acmicpc.net

     

     

    풀이

    풀이가 O(N+N)과 O(N) 2가지가 존재합니다. 하지만 파이썬은 O(N)만 통과합니다.

     

    C++ 풀이 O(N+N)

    문제를 보자마자 대부분의 사람들이 왼쪽에서 오른쪽으로 확인하는 방법을 찾아볼 것 입니다.

    그래서 → 방향으로 확인하는 방법은 주가의 값(arr[i])을 받고, pair(arr[i], i)로 묶어서 정렬하는 방법입니다.

    이 정렬하는 벡터를 max_val이라고 하고, 인덱스를 거꾸로 n-1부터 확인해주면 max_val[i],first가 arr 값중에서 최댓값될 것이고, arr[i]값이 같을 때 max_val[i].second는 i의 값을 최댓값 순서대로 보여줄 것 입니다.

     

    그러면 for문을 돌려서 max_val[j].second와 i를 비교한 후에 i가 second보다 작으면 답에 차이값을 더해주고, 아니라면 while문을 이용하여 아직 확인하지 않은 배열중에서 최댓값을 찾아주면 됩니다.  이때 아직 확인하지 않은 배열에서 값을 뽑아내야하니 visit 배열이 필요하겠네요.

     

    코드

    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
    #include <iostream>
    #include <cstring>
    #include <vector>
    #include <algorithm>
    using namespace std;
    typedef long long ll;
     
    const int N = 1e6 + 1;
    int t, n, visit[N];
    vector<int> arr;
    vector<pair<intint>> max_val;
     
    int main()
    {
        ios_base::sync_with_stdio(0); cin.tie(0); cout.tie(0);
        
        cin >> t;
        while (t--) {
            cin >> n;
            
            arr.resize(n);
            max_val.resize(n);
            memset(visit, 0sizeof(visit));
     
            for (int i = 0; i < n; i++) {
                cin >> arr[i];
                max_val[i] = make_pair(arr[i], i);
            }
     
            sort(max_val.begin(), max_val.end());
            
            long long ans = 0;
            int j = n - 1;
     
            for (int i = 0; i < n; i++) {
                visit[i] = 1;
                if (max_val[j].second != i) ans += max_val[j].first - arr[i];
                else {
                    while (j >= 0 && visit[max_val[j].second]) j--;
                }
            }
     
            cout << ans << "\n";
        }
     
        return 0;
    }
    cs

     

     

    Python 3 풀이 O(N)

    이번엔 관점을 달리하여 ←방향으로 풀어봅시다. 이때는 그냥 val = arr[-1]이라고 잡고, val >= arr[i]면 답에 차이값만큼 더해주고, val < arr[i]이면 val = arr[i]로 업데이트해주면서 for문을 돌리면 됩니다. 그러면 항상 val이 최대값임이 보장이 되니까 O(N+N) 풀이보다 더 간단하게 풀 수 있습니다.

     

    코드

    fast input을 사용하지 않으면 시간 초과가 나옵니다.

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    input = __import__('sys').stdin.readline
     
    for _ in range(int(input())):
        n = int(input())
        arr = [*map(int, input().split())]
     
        ans = 0
        val = arr[-1]
        for i in range(n-1-1-1):
            if arr[i] <= val: ans += val - arr[i]
            else: val = arr[i]
        print(ans)
    cs
    반응형

    '알고리즘 > BOJ' 카테고리의 다른 글

    [BOJ 18310, Python 3] 안테나  (0) 2021.08.03
    [BOJ 11508, Python 3] 2+1 세일  (0) 2021.08.03
    [BOJ 11399, Python 3] ATM  (0) 2021.08.02
    [BOJ 2012, Python 3] 등수 매기기  (0) 2021.08.02
    [BOJ 1449, Python 3] 수리공 항승  (0) 2021.08.02

    댓글

Designed by Tistory.