반응형

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

 

2629번: 양팔저울

첫째 줄에는 추의 개수가 자연수로 주어진다. 추의 개수는 30 이하이다. 둘째 줄에는 추의 무게들이 자연수로 가벼운 것부터 차례로 주어진다. 같은 무게의 추가 여러 개 있을 수도 있다. 추의 무

www.acmicpc.net

 

배낭 문제와 비슷하게 다이나믹 프로그래밍을 이용하여 풀이하였습니다.

 

먼저 주어진 추들을 이용하여 만들 수 있는 모든 무게 조합을 구해줍니다

다음과 같은 예제 입력을 예시로 들면,

4
2 3 3 3
3
1 4 10

만들 수 있는 모든 무게 조합은 [0, 2, 3, 5, 6, 8, 9, 11]이 됩니다.

이를 구하는 소스코드는 다음과 같습니다.

dp[0] = true;
for (int i = 0; i < n; i++)
	for (int j = B_MAX; j >= weight[i]; j--)
		dp[j] |= dp[j - weight[i]];

중복으로 적용되는 것을 방지하기 위하여 역순으로 반복문을 수행해줍니다.

 

 

모든 추들의 무게 조합을 구했으면, 이제 무게를 확인할 수 있는 구슬의 무게들을 구해야합니다.

 

일단 지금 구한 추 조합의 무게를 가지는 구슬들은 모두 확인할 수 있습니다.

추 조합은, 양팔저울의 한쪽에 모두 올라가있는 무게라고 볼 수 있습니다.

그렇다면, 다른 한쪽에는 구슬 1개와 추 0~x개를 함께 올릴 수 있습니다.

 

즉, 추들만 올라가있는 한쪽에서 추를 일부 다른 한쪽으로 옮겨둔 모든 상황에 수평을 이룰 수 있는 구슬의 무게를 생각하면, 무게를 확인할 수 있는 모든 구슬의 무게들을 구할 수 있습니다.

 

위 예시로 판단해보면,

한 쪽에 추들이 5만큼 올라가있다면, 그 쪽에서 임의의 추 몇 개를 다른 한 쪽으로 옮겨줍니다.

만약 무게 2의 추를 옮겼다면, 양 쪽의 무게는 각각 3과 2가 됩니다.

따라서, 양 쪽의 무게 차이는 무게 1의 구슬을 올리면 수평을 맞출 수 있습니다.

 

한쪽에 추들이 8만큼 올라가있다면, 그 쪽에서 임의의 추 몇 개를 다른 한 쪽으로 옮겨줍니다.

만약 무게 3의 추와 무게 3의 추를 옮겼다면, 양 쪽의 무게는 각각 6과 2가 됩니다.

따라서, 양 쪽의 무게 차이는 무게 4의 구슬을 올리면 수평을 맞출 수 있습니다.

 

추의 개수는 30개이고, 추의 최대 무게는 500이기 때문에, 모든 추 무게의 합은 15000이 됩니다.

이를 이용하여 무게 15000까지 만들어낼 수 있는 추 조합을 검사하며,

한 쪽에 올려져있는 추 조합의 일부를 다른 한 쪽으로 옮겼을 때의, 양 쪽의 무게 차이를 구해주었습니다.

그렇다면, 이 모든 무게 차이가 확인할 수 있는 구슬의 무게가 됩니다. 

for (int i = 0; i <= NW_MAX; i++) {
	if (!dp[i]) continue;
	for (int j = B_MAX; j >= i; j--) {
		if (!dp[j]) continue;
		chk[j - i] = chk[j] = true;
	}
}

1. 추들이 한쪽에 다 올려져있을 때, 이미 확인할 수 있는 무게 j

2. j에서 무게 i만큼을 가지는 추의 일부를 다른 한쪽으로 옮겼을 때, 양 쪽의 무게차이 j - i

결국 이 두 가지가 확인 할 수 있는 구슬의 무게입니다.

 

 

전체 소스코드는 다음과 같습니다.

#include <iostream>
#define N_MAX 30
#define W_MAX 500
#define NW_MAX N_MAX * W_MAX
#define B_MAX 40000
using namespace std;

int n, m, b, weight[N_MAX];
bool dp[B_MAX + 1];
bool chk[B_MAX + 1];

int main() {
	ios::sync_with_stdio(false);
	cin.tie(NULL); cout.tie(NULL);

	cin >> n;
	for (int i = 0; i < n; i++)
		cin >> weight[i];

	dp[0] = true;
	for (int i = 0; i < n; i++)
		for (int j = B_MAX; j >= weight[i]; j--)
			dp[j] |= dp[j - weight[i]];
	
	for (int i = 0; i <= NW_MAX; i++) {
		if (!dp[i]) continue;
		for (int j = B_MAX; j >= i; j--) {
			if (!dp[j]) continue;
			chk[j - i] = chk[j] = true;
		}
	}

	cin >> m;
	while (m--) {
		cin >> b;
		cout << (chk[b] ? 'Y' : 'N') << " ";
	}
}

 

 

다른 분들의 간단한 풀이와 소요 시간을 보니 그닥 좋은 풀이는 아닌 것 같습니다.

(모든 추 무게의 합 15000) * (구슬의 무게 40000) 만큼 반복을 돌려야하는게 너무 많은 시간이 걸려서,

set을 이용하여 기억된 무게 조합만 검사하였습니다.

#include <iostream>
#include <unordered_set>
#define N_MAX 30
#define W_MAX 500
#define NW_MAX N_MAX * W_MAX
#define B_MAX 40000
using namespace std;

int n, m, b, weight[N_MAX];
bool dp[B_MAX + 1];
bool chk[B_MAX + 1];

int main() {
	ios::sync_with_stdio(false);
	cin.tie(NULL); cout.tie(NULL);

	cin >> n;
	for (int i = 0; i < n; i++)
		cin >> weight[i];

	unordered_set<int> t;
	dp[0] = true;
	for (int i = 0; i < n; i++)
		for (int j = B_MAX; j >= weight[i]; j--) {
			dp[j] |= dp[j - weight[i]];
			if (dp[j]) t.insert(j);
		}

	for (int i : t) {
		if (i > 15000) continue;
		for(int j : t) {
			if (j - i <= 0) continue;
			chk[j - i] = chk[j] = true;
		}
	}

	cin >> m;
	while (m--) {
		cin >> b;
		cout << (chk[b] ? 'Y' : 'N') << " ";
	}
}

 

속도 개선된 이미지

0ms가 나오는 풀이들도 많지만.. 제 풀이에서만큼은 약간의 개선점을 가지게 되었습니다.

 

 

중복을 제거하려고 set을 썼던건데, 다시 생각해보니 굳이 중복을 만들 필요는 없었네요.

#include <iostream>
#include <vector>
#define N_MAX 30
#define W_MAX 500
#define NW_MAX N_MAX * W_MAX
#define B_MAX 40000
using namespace std;

int n, m, b, weight[N_MAX];
bool dp[B_MAX + 1];
bool chk[B_MAX + 1];
vector<int> t;

int main() {
	ios::sync_with_stdio(false);
	cin.tie(NULL); cout.tie(NULL);

	cin >> n;
	for (int i = 0; i < n; i++)
		cin >> weight[i];

	dp[0] = true;
	for (int i = 0; i < n; i++) {
		for (int j = B_MAX; j >= weight[i]; j--) {
			if (!dp[j] && dp[j - weight[i]]) {
				dp[j] = true;
				t.push_back(j);
			}
		}
	}

	for (int i : t) {
		if (i > 15000) continue;
		for(int j : t) {
			if (j - i <= 0) continue;
			chk[j - i] = chk[j] = true;
		}
	}

	cin >> m;
	while (m--) {
		cin >> b;
		cout << (chk[b] ? 'Y' : 'N') << " ";
	}
}

 

 

속도 개선 이미지

이번에도 약간의 개선점을 가지게 되었습니다.

반응형

'Algorithm' 카테고리의 다른 글

백준 16968 : 차량 번호판 1  (0) 2022.01.30
백준 1781 : 컵라면  (0) 2021.12.25
백준 1162 : 도로포장  (0) 2021.11.23
백준 11780 : 플로이드 2  (0) 2021.11.23
백준 16064 : Coolest Ski Route  (0) 2021.11.22

+ Recent posts