문제 링크
https://www.acmicpc.net/problem/25239
문제
파풀라투스는 메이플스토리의 보스입니다.
파풀라투스가 사용하는 패턴 중, 차원의 균열 봉인 패턴이 있습니다. 이 패턴이 시전되면, 6개의 영역으로 나뉘어진 파풀라투스 시계가 나타나게 됩니다.
[그림 1] 파풀라투스의 시계
시침은 12시 방향에서 2시 방향 사이를 가리킬 때 1번 영역에, 2시에서 4시 사이를 가리킬 때 2번 영역에, 4시 방향에서 6시 방향 사이를 가리킬 때 3번 영역에, 6시 방향에서 8시 방향 사이를 가리킬 때 4번 영역에, 8시 방향에서 10시 방향 사이를 가리킬 때 5번 영역에, 10시 방향에서 12시 방향 사이를 가리킬 때 6번 영역에 있습니다. 시침이 정확히 2, 4, 6, 8, 10, 12시를 가리키는 경우는 고려하지 않습니다.
그림 1에서 시침은 6번 영역에 있습니다. 파풀라투스의 시계는 약 11시 55분이고, 이때 시침은 10시 방향과 12시 방향 사이를 가리키고 있기 때문입니다.
[그림 2] 파풀라투스의 시계가 가리키는 시각을 바꾸는 아이템들
아이템을 먹어서 파풀라투스의 시계가 가리키는 시각을 특정 시간만큼 증가시킬 수 있습니다. 예를 들어, 파풀라투스의 시계가 가리키는 시각이 11시 55분일 때, 10MIN 아이템을 먹는다면, 파풀라투스의 시계가 가리키는 시각은 11시 55분에서 10분 증가한 12시 05분이 됩니다.
[그림 3] 1번, 2번, 5번, 6번 영역이 봉인된 시계
시침이 x번 영역에 있을 때, 유저가 시계 왼쪽 포탈에서 윗키를 누르면, x번 영역이 봉인됩니다. 차원의 균열 패턴이 시전되고 1분이 지난 후, 봉인되지 않은 칸이 있다면, 그 칸들에 쓰여 있는 % 수치의 합만큼 체력을 회복하게 됩니다. 이때, % 수치의 합이 100을 넘어간다면 100%만큼 회복합니다. 만약, 1분이 지나기 전에 6개의 칸이 모두 봉인되었다면, 남은 시간과 관계 없이 패턴이 종료됩니다.
차원의 균열 패턴이 시전되고 난 후 가희가 플레이한 이벤트가 주어졌을 때, 패턴이 끝나고 파풀라투스는 몇 %의 체력을 회복할 수 있는지 구해 주세요.
입력
첫 번째 줄에 차원의 균열 패턴이 시전되었을 때, 파풀라투스의 시계가 가리키고 있는 시각이 hh:mm 형식으로 주어집니다.
두 번째 줄에는 1번 영역에 써져 있는 수부터 6번 영역에 써져 있는 수까지 공백으로 구분되어 주어집니다.
세 번째 줄에 이벤트의 수 L이 주어집니다.
다음 L개의 줄에는 발생한 이벤트 목록이 시간 순서대로 주어집니다. 이벤트 목록에 대한 형식은 아래와 같습니다.
- s.T ^ : 게임 시간 s.T초에 가희가 시계 왼쪽 포탈에서 키보드 위쪽 키를 누릅니다.
- s.T 10MIN : 게임 시간 s.T초에 가희가 10MIN 아이템을 먹습니다.
- s.T 30MIN : 게임 시간 s.T초에 가희가 30MIN 아이템을 먹습니다.
- s.T 50MIN : 게임 시간 s.T초에 가희가 50MIN 아이템을 먹습니다.
- s.T 2HOUR : 게임 시간 s.T초에 가희가 2HOUR 아이템을 먹습니다.
- s.T 4HOUR : 게임 시간 s.T초에 가희가 4HOUR 아이템을 먹습니다.
- s.T 9HOUR : 게임 시간 s.T초에 가희가 9HOUR 아이템을 먹습니다.
s.T 포맷에서 s는 0 이상의 정수로 주어지며, T는 0 이상 999 이하의 정수로 주어집니다. 이때 T가 두 자릿수라면 T 앞에 0 이, 한 자릿수라면 T 앞에 00 이 붙습니다. 이는, 게임 시간 s+T×10-3초를 의미합니다.
출력
차원의 균열 패턴이 끝난 후, 파풀라투스가 회복하는 체력이 h%라고 할 때, h를 출력해 주세요.
제한
- 1 ≤ L ≤ 150
- 차원의 균열 봉인 패턴 외에 다른 패턴들 (집게 패턴, 레이저 패턴 등)은 무시합니다.
- hh:mm 형식에서 hh는 0보다 크거나 같고 11보다 작거나 같은 정수이고, mm은 0보다 크거나 같고 59보다 작거나 같은 정수입니다. hh, mm은 10보다 작은 경우 앞에 0이 붙으며, mm은 10의 배수가 아닙니다.
- 같은 시각에 둘 이상의 이벤트가 주어지지 않으며, 이벤트는 발생한 시간 순으로 주어집니다.
- 차원의 균열 봉인 패턴이 끝난 이후의 이벤트는 주어지지 않습니다.
- 파풀라투스 시계의 칸에 쓰여 있는 수는 1, 10, 100 중 하나입니다.
예제 입력 1
10:13
10 10 10 10 10 100
1
0.101 ^
예제 출력 1
50
파풀라투스의 시계는 처음에 10시 13분을 가리키고 있습니다. 따라서 시침은 6번 영역에 있습니다.
0.101초에 가희가 시계 왼쪽 포탈에서 윗키를 눌러서 6번 영역이 봉인되게 됩니다. 그 이후에 가희가 아이템을 먹거나 영역을 봉인시키는 이벤트가 없으므로, 차원의 균열 패턴이 끝난 후에 봉인되지 않는 1번, 2번, 3번, 4번, 5번 영역에 쓰여져 있는 수의 합인 50%만큼 파풀라투스가 체력을 회복하게 됩니다.
예제 입력 2
10:13
1 10 10 10 10 100
3
0.101 ^
2.333 2HOUR
4.444 ^
예제 출력 2
40
파풀라투스의 시계는 처음에 10시 13분을 가리키고 있습니다. 따라서 시침은 6번 영역에 있습니다.
0.101초에 가희가 시계 왼쪽 포탈에서 윗키를 눌러서 6번 영역이 봉인되게 됩니다. 2.333초에 가희가 2HOUR 아이템을 먹었기 때문에, 파풀라투스의 시계는 00시 13분을 가리키게 됩니다. 이때, 시침은 1번 영역에 있습니다. 4.444초에 가희가 시계 왼쪽 포탈에서 윗키를 눌렀기 때문에, 1번 영역이 봉인되게 됩니다. 그 이후에 가희가 아이템을 먹거나 영역을 봉인시키는 이벤트가 없으므로, 차원의 균열 패턴이 끝난 후에 봉인되지 않는 2번, 3번, 4번, 5번 영역에 쓰여져 있는 수의 합인 40%만큼 파풀라투스가 체력을 회복하게 됩니다.
알고리즘 분류
- 구현
- 문자열
풀이
처음에 파풀라투스의 시계가 가리키는 시각에 따라 시침이 어느 영역에 있는지를 판단하는데, 이것을 분으로 따진다.
예를 들면 10시 13분을 가리키고 있다면, 613분이며, 이 때 파풀라투스의 시침은 6번 영역에 있는 것이다.
그리고 다음 L번의 이벤트를 처리한다.
s.T초에 ^라면 현재 영역이 봉인된 상태가 아닌 경우, 영역을 봉인시킨다. ^가 아니라면, 아이템을 먹은 것인데, 예를 들어 10MIN이라면 10분짜리 아이템을 먹은 것이다. 따라서 10시 23분, 즉 623분이 된다. 즉 먹은 아이템에 따라 시간을 변화시키면 되는 것이다.
하지만 문제를 잘 읽어보면, 차원의 균열 봉인 패턴의 주어진 제한 시간은 없으므로, s.T초는 중요하지 않다는 것을 알 수 있다. 따라서 s.T 옆에 있는 행동만을 확인해주면 된다.
이벤트를 처리하는 도중 모든 영역이 봉인됐다면 그 즉시 패턴은 종료되고, 파풀라투스는 체력을 회복하지 않으므로 0을 출력한다.
이벤트를 모두 처리한 후 봉인되지 않은 영역에 적혀 있는 파풀라투스가 %단위로 회복할 체력의 합을 구한다. 그 합이 100을 넘어갔다면 100을 출력하고, 100 이하라면 그대로 그 회복할 체력의 합을 출력한다.
코드
#include <iostream>
#include <string>
#include <vector>
#include <algorithm>
#define FASTIO cin.tie(NULL); cout.tie(NULL); ios::sync_with_stdio(false);
using namespace std;
string Cur;
int When, Where;
int Place[7];
bool Clear[7];
int L;
string Time, Play;
int Answer = 0;
void Relocate() {
if ((When > 0) && (When < 120)) {
Where = 1;
}
else if ((When > 120) && (When < 240)) {
Where = 2;
}
else if ((When > 240) && (When < 360)) {
Where = 3;
}
else if ((When > 360) && (When < 480)) {
Where = 4;
}
else if ((When > 480) && (When < 600)) {
Where = 5;
}
else if ((When > 600) && (When < 720)) {
Where = 6;
}
}
void Init() {
int Hour = stoi(Cur.substr(0, 2));
int Minute = stoi(Cur.substr(3));
When = (Hour * 60) + Minute;
Relocate();
}
bool All_Clear() {
for (int i = 1; i <= 6; i++) {
if (!Clear[i]) {
return false;
}
}
return true;
}
void Settings() {
if (Play == "^") {
if (!Clear[Where]) {
Clear[Where] = true;
}
}
else if (Play == "10MIN") {
When += 10;
}
else if (Play == "30MIN") {
When += 30;
}
else if (Play == "50MIN") {
When += 50;
}
else if (Play == "2HOUR") {
When += 120;
}
else if (Play == "4HOUR") {
When += 240;
}
else if (Play == "9HOUR") {
When += 540;
}
if (When >= 720) {
When -= 720;
}
Relocate();
}
void Input() {
cin >> Cur;
Init();
for (int i = 1; i <= 6; i++) {
cin >> Place[i];
}
cin >> L;
for (int i = 0; i < L; i++) {
cin >> Time >> Play;
Settings();
if (All_Clear()) {
break;
}
}
}
int Heal() {
int Res = 0;
for (int i = 1; i <= 6; i++) {
if (!Clear[i]) {
Res += Place[i];
}
}
if (Res < 100) {
return Res;
}
else {
return 100;
}
}
void Find_Answer() {
cout << Heal() << "\n";
}
int main() {
FASTIO
Input();
Settings();
Find_Answer();
return 0;
}
'BOJ > Silver' 카테고리의 다른 글
[BOJ/Silver 3] 백준 9613 GCD 합(C++) (0) | 2022.06.08 |
---|---|
[BOJ/Silver 3] 백준 1735 분수 합(C++) (0) | 2022.06.08 |
[BOJ/Silver 1] 백준 2730 오늘은 OS 숙제 제출일(C++) (0) | 2022.05.27 |
[BOJ/Silver 3] 백준 11332 시간초과(C++) (0) | 2022.05.26 |
[BOJ/Silver 5] 백준 21966 (중략)(C++) (0) | 2022.05.26 |