티스토리 뷰

BOJ

[BOJ] 1084 방 번호 2

[gunwookim] 2020. 4. 13. 22:04

알고리즘은 그리디 입니다.

일단 제일 가격이 낮은 숫자들로 최대한 채웁니다.

이때, 0 이 제일 싼 가격이면 그 다음으로 싼 가격의 숫자를 맨 앞에 놓고 그 뒤에 0을 쭉 붙입니다.

그러면 쓰고 남는 돈이 있는데 이 돈으로 최대한 큰 수로 바꿉니다.

각 숫자에 대해 몇개씩 살 수 있는지 나오는데 이 숫자들을 내림차순으로 나열하면 제일 큰 숫자를 만들 수 있습니다.

제가 쫌 코드를 드럽게 짜기 때문에 풀이 이해용으로만 보시기 바랍니다. 

 

#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
int n,pl,End;
ll k,x,co[15],ch[15],tmp[15];
 
struct Money {
    ll cost; int num;
}a[15];
 
bool cmp(Money p1,Money p2) {
    if(p1.cost == p2.cost) return p1.num > p2.num;
    return p1.cost < p2.cost;
}
 
int printLen() {
    End = pl = 0;
    if(k < a[1].cost) return 1;
    if(a[1].num == 0) {
        if(k < a[2].cost) return 2;
        k -= a[2].cost; pl = 1;
    }
    ch[a[1].num] = k/a[1].cost;
    ch[a[2].num] = pl;
    cout << k/a[1].cost+pl << '\n';
    k %= a[1].cost;
    return 0;
}
 
void Solve() {
    ll diff;
    for(int i = n-1;i > a[1].num;i--) {
        diff = co[i]-a[1].cost;
        ll change = min(ch[a[1].num],k/diff);
        ch[i] += change;
        ch[a[1].num] -= change;
        k %= diff;
    }
    for(int i = 0;i < n;i++) tmp[i] = ch[i];
    for(int i = n-1,go = 1;i >= 0;i--) {
        while(go <= 50&&ch[i]) {
            cout << i;
            go++; ch[i]--;
        }
    }
    cout << '\n';
    for(int i = 0;i < n;i++) ch[i] = tmp[i];
    vector <int> ans;
    for(int i = 0,go = 1;i < n;i++) {
        while(go <= 50&&ch[i]) {
            ans.push_back(i);
            go++; ch[i]--;
        }
    }
    reverse(ans.begin(),ans.end());
    for(int i : ans) cout << i;
    cout << '\n';
}
 
int main() {
    ios_base::sync_with_stdio(false); cin.tie(0);
    while(cin >> n) {
        memset(ch,0,sizeof(ch));
        if(n == 1) {
            cin >> x >> k;
            if(x > k) cout << "0\n\n\n";
            else cout << "1\n0\n0\n";
            continue;
        }
        for(int i = 1;i <= n;i++) {
            cin >> co[i-1];
            a[i] = {co[i-1],i-1};
        }
        cin >> k;
        sort(a+1,a+n+1,cmp);
        x = printLen();
        if(x == 1) { cout << "0\n\n\n"; continue; } // 아무것도 안되는 경우
        if(x == 2) { cout << "1\n0\n0\n"; continue; } // 0만 되는 경우
        if(pl) {
            for(int i = n-1;i > a[2].num;i--) {
                if(k >= co[i]-a[2].cost) {
                    k -= co[i]-a[2].cost;
                    ch[a[2].num]--; ch[i]++;
                    break;
                }
            }
        }
           Solve();
    }
}

'BOJ' 카테고리의 다른 글

[BOJ] 17302 흰색으로 만들기  (0) 2020.05.21
[BOJ] 2473 세 용액  (2) 2020.04.13
[BOJ] 13536 수열과 쿼리 4  (0) 2020.04.13
[BOJ] 18291 비요뜨의 징검다리 건너기  (0) 2020.04.13
[BOJ] 18830 하이퍼 수열과 하이퍼 쿼리  (0) 2020.04.13
댓글
공지사항
최근에 올라온 글
최근에 달린 댓글
Total
Today
Yesterday
링크
«   2025/01   »
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
글 보관함