알고리즘/[ Baekjoon ]

[ BOJ ][JAVA][20546] 기적의 매매법

kim.svadoz 2021. 5. 11. 22:32
반응형

www.acmicpc.net/problem/20546

 

20546번: 🐜 기적의 매매법 🐜

1월 14일 기준 준현이의 자산이 더 크다면 "BNP"를, 성민이의 자산이 더 크다면 "TIMING"을 출력한다. 둘의 자산이 같다면 "SAMESAME"을 출력한다. 모든 결과 따옴표를 제외하고 출력한다.

www.acmicpc.net

시간 제한 메모리 제한 제출 정답 맞은 사람 정답 비율
1 초 512 MB 287 170 151 58.755%

문제

"오늘도 호재만 있게 해주세요. 버핏-"

2년차 개미 준현이는 오늘도 버핏신에게 기도를 올린다. 장기 투자를 지향하는 준현이는 한 번 산 주식은 절대 팔지 않는다. 2099년이 되어도 주식을 팔지 않을 것이다. 주식 매수 후 오로지 기도만 하기 때문에 이를 BNP 전략이라고 한다. BNP는 Buy and Pray의 약자이다. 준현이는 주식을 살 수 있다면 무조건 최대한 많이 산다. 준현이는 욕심쟁이이기 때문에, 주식을 살 수 있다면 가능한 만큼 즉시 매수한다. 다음은 준현이가 현금 100원으로 A기업의 주식을 사는 경우이다.

  1일 2일 3일 4일 5일 6일 7일
현금 100 20 20 6 0 0 0
주가 40 33 7 2 1 12 50
매수 가능 주식 수 2 0 2 3 0 0 0
매수여부 O X O O X X X
남은 현금 20 20 6 0 0 0 0
보유 주식 수 2 2 4 7 7 7 7

"주식은 타이밍이지!"

반면, 성민이는 오늘도 주식 거래 프로그램을 실행한다. 모니터 8개에서 뿜어져 나오는 화려한 주식 차트가 성민이를 감싼다. 성민이는 주식이 타이밍 싸움이라 생각한다. 전형적인 단기 투자자로 생각하면 오산이다. 성민이만의 전략이 있기 때문이다. 이른바 33 매매법으로, 그 방법은 다음의 세 가지 룰로 이루어져있다.

  • 모든 거래는 전량 매수와 전량 매도로 이루어진다. 현재 가지고 있는 현금이 100원이고 주가가 11원이라면 99원어치의 주식을 매수하는 것이다. 단, 현금이 100원 있고 주가가 101원이라면 주식을 살 수 없다. 성민이는 빚을 내서 주식을 하지는 않는다.
  • 3일 연속 가격이 전일 대비 상승하는 주식은 다음날 무조건 가격이 하락한다고 가정한다. 따라서 현재 소유한 주식의 가격이 3일째 상승한다면, 전량 매도한다. 전일과 오늘의 주가가 동일하다면 가격이 상승한 것이 아니다.
  • 3일 연속 가격이 전일 대비 하락하는 주식은 다음날 무조건 가격이 상승한다고 가정한다. 따라서 이러한 경향이 나타나면 즉시 주식을 전량 매수한다. 전일과 오늘의 주가가 동일하다면 가격이 하락한 것이 아니다.

준현이와 성민이는 각자의 매매법 중 어떤 방법이 더 수익률이 높은지 겨뤄보기로 했다. 오로지 MachineDuck이라는 기업의 주식만 거래가 가능하며, 내기 기간은 2021년 1월 1일부터 2021년 1월 14일까지이다. 준현이와 성민이에게 주어진 현금은 동일하다. 세기의 대결이기 때문에 이 기간에는 매일 주식 거래가 가능하다. 2021년 1월 14일에 더 많은 자산을 보유한 사람이 승리한다. 1월 14일의 자산은 (현금 + 1월 14일의 주가 × 주식 수)로 계산한다.

우리는 2021년 1월 1일부터 2021년 1월 14일까지의 주식 가격을 미리 알고 있다. 준현이와 성민이 중 누가 더 높은 수익률을 낼지 맞혀보자!

입력

첫 번째 줄에 준현이와 성민이에게 주어진 현금이 주어진다.

두 번째 줄에 2021년 1월 1일부터 2021년 1월 14일까지의 MachineDuck 주가가 공백을 두고 차례대로 주어진다. 모든 입력은 1000 이하의 양의 정수이다.

출력

1월 14일 기준 준현이의 자산이 더 크다면 "BNP"를, 성민이의 자산이 더 크다면 "TIMING"을 출력한다.

둘의 자산이 같다면 "SAMESAME"을 출력한다. 모든 결과 따옴표를 제외하고 출력한다.

예제 입력 1

100
10 20 23 34 55 30 22 19 12 45 23 44 34 38

예제 출력 1

BNP

준현이는 1월 1일에 10주를 매수한다. 따라서 1월 14일 380원의 자산을 가지게 된다.

성민이는 1월 8일에 5주를 매수한다. 따라서 1월 14일 195원의 자산을 가지게 된다.

예제 입력 2

15
20 20 33 98 15 6 4 1 1 1 2 3 6 14

예제 출력 2

TIMING

코드

import java.io.*;
import java.util.*;

public class p20546 {
    static class Stock {
        int cash, cnt;
        public Stock(int cash, int cnt) {
            this.cash = cash;
            this.cnt = cnt;
        }

        void buyStock(int todayCost, int cnt) {
            if (this.cash < todayCost * cnt) return;
            this.cnt += cnt;
            this.cash -= todayCost * cnt;
        }

        void sellStock(int todayCost) {
            this.cash += todayCost * this.cnt;
            this.cnt = 0;
        }

    }
    static int n;
    static int[] price;
    public static void main(String[] args) throws IOException {
        BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
        n = Integer.parseInt(br.readLine());
        StringTokenizer st= new StringTokenizer(br.readLine());
        price = new int[14];
        for (int i = 0; i < 14; i++) {
            price[i] = Integer.parseInt(st.nextToken());
        }

        int j = junhyun();
        int s = sungmin();
        if (j > s) {
            System.out.println("BNP");
        } else if (j == s) {
            System.out.println("SAMESAME");
        } else {
            System.out.println("TIMING");
        }
    }
    static int junhyun() {
        Stock Stk = new Stock(n, 0);
        for (int today : price) {
            int getCash = Stk.cash;
            int buyCnt = getCash / today;

            // 전량 풀매수
            if (buyCnt > 0){
                Stk.buyStock(today, buyCnt);
            }
        }
        int ret = Stk.cash + (Stk.cnt * price[13]);
        return ret;
    }

    static int sungmin() {
        Stock Stk = new Stock(n, 0);
        for (int i = 3; i < 14; i++) {
            int today = price[i];
            int getCash = Stk.cash;
            int buyCnt = getCash / today;

            // 점점 상승 -> 풀매도
            if (price[i - 1] > price[i - 2] && price[i - 2] > price[i - 3]) {
                Stk.sellStock(today);
            } else if (price[i - 1] < price[i - 2] && price[i - 2] < price[i - 3]) { // 풀매수
                Stk.buyStock(today, buyCnt);
            }
        }
        int ret = Stk.cash + (Stk.cnt * price[13]);
        return ret;
    }

}
반응형