문제 링크
https://www.acmicpc.net/problem/21776
문제
가희와 친구들은 읽기 쓰기 놀이를 하고 있습니다. 읽기 쓰기 놀이는 문자열을 가지고 시작합니다.
놀이에서 사용하는 카드에 적혀져 있는 연산은 둘 중 하나입니다.
- add c
- 문자열 뒤에 문자 c를 추가합니다.
- del x
- 문자열의 x번째 위치에 있는 글자를 삭제합니다.
- 문자열의 인덱스는 0부터 시작합니다. x번째 위치에 있는 문자를 삭제할 수 없는 경우에는, 오류가 발생합니다.
놀이의 규칙은 다음과 같습니다.
- 빈 문자열로 게임을 시작합니다.
- 각 턴을 수행하는 사람은 1명입니다.
- 턴을 수행하는 사람은 가지고 있는 카드에 적혀져 있는 연산을 모두 수행하고 턴을 종료합니다. 턴을 수행하다가, 오류가 발생하면 문자열은 "ERROR"가 되고, 즉시 게임이 종료됩니다.
- 게임의 끝났을 때, 문자열이 빈 문자열이라면, 문자열은 "EMPTY"가 됩니다.
문자열 게임에 참가하는 사람은 N명이고, 카드는 C장 있습니다.
게임에 참가하는 사람이 어떤 순서대로 카드를 냈는지 알고 있을 때, 게임의 결과로 나올 수 있는 문자열을 사전순으로 출력해 주세요.
입력
1번째 줄에 N, C가 공백으로 구분되어 주어집니다.
2번째 줄 부터 N+1번째 줄까지 1번 사람부터 N번 사람까지 낸 카드의 갯수와 카드를 낸 순서가 주어집니다.
예를 들어 3번째 줄에 3 2 4 5 가 있다면, 2번째 사람이 3개의 카드 2,4,5를 순서대로 낸 것을 의미합니다.
N+2번째 줄부터 N+C+1번째 줄까지 1번 카드부터 C번 카드에 적혀져 있는 1개 이상의 연산이 주어집니다.
연산이 여러 개 있는 경우에 각각의 연산은 ,으로 구분되어 주어집니다.
출력
게임의 결과로 나올 수 있는 문자열을 사전순으로 출력해 주세요. 사전순의 기준은 아스키 코드입니다.
만약에 같은 문자열이 여러 개가 나오면 하나로 출력해야 합니다.
제한
- 1 ≤ N ≤ C ≤ 9
- 1 ≤ 카드 C개에 있는 연산 갯수 합 ≤ 10
- 추가되는 문자는 소문자입니다.
- 0 ≤ 제거 연산에서 등장하는 수 ≤ 9
- 카드에는 하나 이상의 연산이 있습니다.
- 모든 플레이어는 최소 한 장의 카드를 냅니다.
- 모든 카드는 게임에 이용되며, 한 번 사용된 카드는 다시 사용되지 않습니다.
예제 입력 1
2 2
1 1
1 2
ADD a,ADD a,ADD d
DEL 0
예제 출력 1
ERROR
ad
가능한 가짓수에 대한 설명은 다음과 같습니다.
- 1번 사람이 먼저 1번 카드에 있는 연산들을 모두 수행한 후에 문자열은 "aad"가 됩니다.
다음에 2번이 2번 카드에 있는 연산을 모두 수행한 후에는, "aad"의 0번째 원소인 "a"가 지워집니다. 최종적으로 "ad"가 됩니다. - 2번 사람이 먼저 2번 카드 연산을 수행할 경우. 문자열은 처음에 비어 있습니다. 비어 있는 문자열의 0번째 원소에 접근을 시도하므로, 에러입니다.
아래의 플레이는 턴을 수행하는 사람은 카드에 적혀져 있는 모든 명령을 수행하고 턴을 종료한다는 조건을 만족하지 않습니다.
- 1번은 1번 카드에 있는 ADD a 연산을 수행합니다.
- 2번은 2번 카드에 있는 DEL 0 연산을 수행합니다.
- 1번은 1번 카드에 있는 ADD a 연산과 ADD d 연산을 수행합니다.
예제 입력 2
2 3
2 1 2
1 3
ADD a
ADD b
ADD c
예제 출력 2
abc
acb
cab
예제 입력 3
2 2
1 1
1 2
DEL 0
DEL 0
예제 출력 3
ERROR
가능한 가짓수에 대한 설명은 아래와 같습니다.
- 2번이 먼저 DEL 0을 수행하려는 경우 빈 문자열에 DEL 0을 수행하므로 ERROR입니다.
- 1번이 먼저 DEL 0을 수행하려는 경우 빈 문자열에 DEL 0을 수행하므로 ERROR입니다.
게임의 결과로 나올 수 있는 문자열은 "ERROR", "ERROR"입니다. "ERROR" 문자열이 중복해서 나왔으므로, "ERROR" 문자열 하나로 출력해야 합니다.
알고리즘 분류
- 구현
- 백트래킹
풀이
카드 C장은 각각 한 장씩 있고, 모든 플에이어는 카드를 순서대로 낸다. 따라서 next_permutation 함수를 사용한다.
예제 2번을 보면 1번 사람은 카드를 1장, 2번 사람은 카드를 2장 가지고 있다. 따라서 처음 순서는 (1, 2, 2)이다. 그러면 카드를 내는 순서는 (3, 1, 2)가 될 것이다.
다음 순서는 (2, 1, 2)이며, 카드를 내는 순서는 (1, 3, 2)가 될 것이다. 이렇게 next_permutation 함수를 사용하여 카드를 낼 수 있는 모든 순서에서 나올 수 있는 문자열을 찾는다.
다만 특정 턴에서 에러가 발생하면 문자열을 즉시 ERROR로 바꾸고 게임을 종료한다.
또한 게임이 종료된 후 문자열이 비어 있다면 문자열을 EMPTY로 바꾼다.
마지막으로 나올 수 있는 문자열들을 담은 벡터를 오름차순 정렬한 후 중복을 제거하고 출력한다.
코드
#include <iostream>
#include <vector>
#include <string>
#include <algorithm>
#define FASTIO cin.tie(0); cout.tie(0);
#define MAX 10
using namespace std;
int N, C;
vector<int> Card_Vec[MAX];
vector<string> Op_Vec;
vector<int> Order_Vec;
vector<string> Answer;
void input() {
cin >> N >> C;
for (int i = 0; i < N; i++) {
int Card_Size;
cin >> Card_Size;
for (int j = 0; j < Card_Size; j++) {
int Card;
cin >> Card;
Card_Vec[i].push_back(Card);
Order_Vec.push_back(i);
}
}
cin.ignore();
for (int i = 0; i < C; i++) {
string Operation;
getline(cin, Operation);
Op_Vec.push_back(Operation);
}
}
string command_Operate(string Command, string S) {
vector<string> Vec;
string tmp = "";
for (int i = 0; i < Command.size(); i++) {
if (Command[i] == ',') {
Vec.push_back(tmp);
tmp = "";
}
else {
tmp += Command[i];
}
}
Vec.push_back(tmp);
tmp = "";
string res = S;
for (int i = 0; i < Vec.size(); i++) {
string cmd = Vec[i];
if (cmd[0] == 'A') {
char C = cmd[4];
res += C;
}
else if (cmd[0] == 'D') {
int X = (cmd[4] - '0');
if (res.size() <= X) {
return "ERROR";
}
res.erase(res.begin() + X);
}
}
return res;
}
void settings() {
do {
string S = "";
int Index[MAX] = { 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 };
for (int i = 0; i < Order_Vec.size(); i++) {
int Player = Order_Vec[i];
int Cur_Card = Card_Vec[Player][Index[Player]++];
S = command_Operate(Op_Vec[Cur_Card - 1], S);
if (S == "ERROR") {
break;
}
}
if (S == "") {
Answer.push_back("EMPTY");
}
else {
Answer.push_back(S);
}
} while (next_permutation(Order_Vec.begin(), Order_Vec.end()));
}
void find_Answer() {
sort(Answer.begin(), Answer.end());
Answer.erase(unique(Answer.begin(), Answer.end()), Answer.end());
for (int i = 0; i < Answer.size(); i++) {
cout << Answer[i] << "\n";
}
}
int main() {
FASTIO
input();
settings();
find_Answer();
return 0;
}
'BOJ > Gold' 카테고리의 다른 글
[BOJ/Gold 4] 백준 12442 약속장소 정하기 (Large)(C++) (0) | 2022.12.02 |
---|---|
[BOJ/Gold 4] 백준 1504 특정한 최단 경로(C++) (0) | 2022.11.30 |
[BOJ/Gold 3] 백준 25240 가희와 파일 탐색기 2(C++) (0) | 2022.10.07 |
[BOJ/Gold 4] 백준 23294 웹 브라우저 1(C++) (0) | 2022.10.07 |
[BOJ/Gold 4] 백준 23031 으어어... 에이쁠 주세요..(C++) (0) | 2022.10.07 |