BOJ - 문자열 게임(18241)


문제(링크)

https://hellogaon.tistory.com/75
위 블로그를 참고하여 풀이했습니다. 구현이 까다로운 문제였습니다.

왼쪽 덱(DL), 오른쪽 덱(DR)을 각각 만들어 그리디(?)하게
L연산 시에는 왼쪽부터 가장 처음 나오는 W문자열을 지워주고,
R연산 시에는 오른쪽부터 가장 처음 나오는 W문자열을 지워줍니다.
(왼쪽, 오른쪽부터 두 개의 포인터를 두어 두 포인터가 겹칠 때까지 진행합니다.)

두 포인터가 겹친다면, DL 엔 왼쪽의 문자열들, DR 엔 오른쪽의 문자열들이 모두 있기 때문에
DL의 오른쪽, DR의 왼쪽을 합쳤을 때 W문자열이 존재할 수 있습니다.

DL의 오른쪽을 pop -> DR의 왼쪽에 push 해가며 W문자열을 만들 수 있는지 조사하면 됩니다.

Perfect 인지 You Lose 인지 여부는 n개의 L, R 연산 뒤에 임의의 연산 L 을 추가하여 해결했습니다.


#include<iostream>
#include<algorithm>
#include<math.h>
#include<string>
#include<vector>
#include<stack>
#include<queue>
#include<map>
using namespace std;
#define FIO ios_base::sync_with_stdio(false); cin.tie(NULL)
#define pii pair<int, int>
#define pdd pair<double, double>
#define pic pair<int, char>
#define ll long long
#define vi vector<int>
#define vl vector<long long>
#define vc vector<char>
#define vii vector<pii>
#define IMAX 2100000001
#define LMAX 1000000000000000000
#define DMAX 0xFFFFFFFFFFFFF
int mv1[4] = { 0, 1, 0, -1 };
int mv2[4] = { -1, 0, 1, 0 };

deque<char> DL;
deque<char> DR;
string W, S;
int n;

int main(void) {
 FIO;

 cin >> W >> S;

 cin >> n;
 
 int l_idx = 0;
 int r_idx = S.size() - 1;

 int cnt = 0;
 int temp_cnt = 0;
 string ans = "";

 for (int i = 0; i < n + 1; i++)
 {
  char q;
  if (i != n)
   cin >> q;
  else {
   temp_cnt = cnt;
   //수행한 연산의 갯수 저장

   for (int i = 0; i < DL.size(); i++)
   {
    ans += DL[i];
   }

   for (int i = l_idx; i <= r_idx; i++)
   {
    ans += S[i];
   }

   for (int i = 0; i < DR.size(); i++)
   {
    ans += DR[i];
   }

   q = 'L';
   //마지막 문자열에 W가 존재하는지 판단하기 위해 임의의 L 연산 추가
  }

  if (l_idx <= r_idx) {
   //겹치지 않는다면
   if (q == 'L') {

    bool find = false;

    while (!find) {
     DL.push_back(S[l_idx++]);

     if (DL.size() < W.size() && l_idx < r_idx) continue;

     int W_idx = 0;
     bool match = true;

     if (DL.size() < W.size())
      match = false;

     for (int i = DL.size() - W.size(); i < DL.size(); i++)
     {
      if ( DL[i] != W[W_idx++]) {
       match = false;
       break;
      }
     }

     if (match) {
      cnt++;
      find = true;

      for (int i = 0; i < W.size(); i++)
      {
       DL.pop_back();
      }
     }

     if (l_idx > r_idx) break;
    }

   }
   else {
    bool find = false;

    while (!find) {
     DR.push_front(S[r_idx--]);

     if (DR.size() < W.size() && l_idx < r_idx) continue;

     bool match = true;

     if (DR.size() < W.size())
      match = false;

     for (int i = 0; i < W.size(); i++)
     {
      if (DR[i] != W[i]) {
       match = false;
       break;
      }
     }

     if (match) {
      cnt++;
      find = true;

      for (int i = 0; i < W.size(); i++)
      {
       DR.pop_front();
      }
     }
     if (l_idx > r_idx) break;
    }
   }
  }

  if (l_idx > r_idx) {
   //두 포인터가 겹쳤을 경우

   while (!DL.empty()) {
    DR.push_front(DL.back()); DL.pop_back();

    bool match = true;

    if (DR.size() >= W.size()) {
     for (int i = 0; i < W.size(); i++)
     {
      if (DR[i] != W[i]) {
       match = false;
       break;
      }
     }
     if (!match) continue;


     for (int i = 0; i < W.size(); i++)
     {
      DR.pop_front();
     }

     cnt++;
     break;
    }
   }
  }
 }

 cout << temp_cnt << "\n";

 cout << ans << "\n";

 if (cnt == temp_cnt)  //마지막 연산으로 cnt 값이 증가하면 실패
  cout << "Perfect!";
 else
  cout << "You Lose!";
}

댓글

이 블로그의 인기 게시물

HTML - input file 버튼 꾸미기

HTML - 이미지 미리보기(jQuery 없이)

BOJ - DNA 유사도(2612)