PS (C, C++)

[백준/C++] 1963 소수 경로

최연재 2024. 10. 2. 23:06

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

 

코드

#include <iostream>
#include <queue>
#include <cstring>
using namespace std;

#define MAX 10001
#define FASTIO ios::sync_with_stdio(false); cin.tie(NULL);

typedef pair<int, int> pi;
bool isPrime[MAX], visit[MAX];
int change[4] = { 1000, 100, 10, 1 };

void checkPrime() {
	memset(isPrime, true, sizeof(isPrime));
	for (int i = 2; i*i<MAX; i++) {
		if (isPrime[i]) {
			for (int j = i * i; j < MAX; j += i)
				isPrime[j] = false;
		}
	}
}

void bfs(int now, int goal) {
	queue<pi> q;

	q.push({ now, 0 });
	visit[now] = true;

	while (!q.empty()) {
		int value = q.front().first;
		int cnt = q.front().second;
		q.pop();

		if (goal == value) {
			cout << cnt << "\n";
			return;
		}

		for (int i = 0, tmp; i < 4; i++) {
			tmp = value;
			int p = (value / change[i]) % 10;
			tmp -= (p * change[i]);

			for (int j = 0; j < 10; j++) {
				if (i == 0 && j == 0) continue;
				int next = tmp + j * change[i];
				if (next >= 1000 && next <= 9999 && !visit[next] && isPrime[next]) {
					q.push({ next, cnt + 1 });
					visit[next] = true;
				}
			}
		}
	}
	cout << "Impossible\n";
}

int main() {
	FASTIO;
	int t, a, b;
	cin >> t;

	checkPrime();
	
	while (t--) {
		cin >> a >> b;
		bfs(a, b);
		memset(visit, false, sizeof(visit));
	}
	return 0;
}

설명

필요한 변수를 입력받은 후 checkPrime 함수로 미리 천의 자리 수까지 소수 판별을 합니다. 이후 테스트케이스마다 현재의 값, 바꾸고자 하는 값을 입력받은 후, 이에 대해서 bfs 함수를 호출합니다. 

 

bfs 함수에서는 기본적으로 큐에  현재의 값과 처음의 값에서 현재 값까지 오는 데 필요한 횟수를 저장해두고 이를 이용합니다.  for문 내에서는 천의 자리부터 일의 자리까지 값을 전부 다 바꿔보면서 천의 자리 소수이면서 방문하지 않은 값에 대해서만 큐에 넣어줍니다.

 

목표하는 값에 도달하였다면 오는 데 걸리는 횟수를 출력하고, 큐가 빌 때까지 도달하지 못했다면 Impossible을 출력하면 됩니다.

 

느낀 점

소수를 에라토스테네스의 체로 미리 판별해둔 후, 자릿수를 하나 바꿨을 때에도 소수일 경우에 대해서만 BFS를 돌리면 됩니다!

 

'PS (C, C++)' 카테고리의 다른 글

[백준/C++] 14923 미로 탈출  (1) 2024.10.03
[백준/C++] 1600 말이 되고픈 원숭이  (0) 2024.10.03
[백준/C++] 2234 성곽  (0) 2024.10.01
[백준/C++] 14382 숫자세는 양 (Large)  (1) 2024.09.25
[백준/C++] 2637 장난감 조립  (0) 2024.09.24