PS (C, C++)

[백준/C++] 17609 회문

최연재 2024. 11. 21. 02:23

https://www.acmicpc.net/problem/17609

코드

#include <iostream>
using namespace std;

#define FASTIO ios::sync_with_stdio(false); cin.tie(NULL);
int t;
string s;

int makePal(int left, int right, bool isUse) {
	while (left < right) {
		if (isUse && s[left] != s[right])
			return 2;

		if (s[left] == s[right]) {
			left++;
			right--;
		}
		else {
			if (makePal(left + 1, right, true) == 0 || makePal(left, right - 1, true) == 0)
				return 1;
			else return 2;
		}
	}
	return 0;
}

int main() {
	FASTIO;
	cin >> t;
	while (t--) {
		cin >> s;
		cout << makePal(0, s.length()-1, false) << "\n";
	}
	return 0;
}

설명

아이디어는 다음과 같다.

 

left는 맨 왼쪽, right는 맨 오른쪽에서 출발해서 가운데로 각자 이동한다. s[left] == s[right]이면 회문 조건을 충족하는 것이니 계속 가운데로 이동시킨다. 만약 s[left] != s[right] 라면 회문이 아니다. 다만 유사회문일 가능성이 있으므로 left를 한 번 더 이동시키거나, right를 한 번 더 이동시켜주고 isUse를 true로 바꾼다. 이때부터 다시 위의 상황을 반복하는데 이미 회문이 아니므로 s[left] != s[right] 이면 일반 문자열이다. 

 

이를 계속 반복해주고 상황에 맞는 반환값을 리턴한다.

 

 

느낀 점

아래처럼 코드를 짜서 한 번 틀렸다.

더보기
    while (t--) {
		cin >> s;
		bool flag = true, getAns = false;
		left = 0, right = s.length() - 1;

		while (left < right) {
			if (!flag && s[left] != s[right]) {
				cout << 2 << "\n";
				getAns = true;
				break;
			}

			if (s[left] == s[right]) {
				left++;
				right--;
			}
			else {
				if (left + 1 < right && s[left + 1] == s[right]) {
					right--;
					left += 2;
					
				}
				else if (right - 1 > left && s[right - 1] == s[left]) {
					left++;
					right -= 2;
				}
				flag = false;
			}
		}

		if (!getAns && flag) cout << 0 << "\n";
		else if (!getAns && !flag) cout << 1 << "\n";
	}

 이유는 투포인터 부분에서 else 내에서 if 와 else if가 같은 상황에서 실행되었어야 하는데 그렇지 못해서였다. 그래서 left, right를 이동시키는 투 포인터 파트 자체를 함수로 뽑아냈다. 

 

이렇게 하니 바로 통과했다! 아이디어 자체는 어렵지 않았던 것 같다.