首页 > 代码库 > uva562 - Dividing coins(01背包)

uva562 - Dividing coins(01背包)

题目:uva562 - Dividing coins(01背包)


题目大意:给出N个硬币,每个硬币有对应的面值。要求将这些硬币分成两部分,求这两部分最小的差值。


解题思路:先求这些硬币能够凑出的钱(0, 1背包),然后再从sum(这些硬币的总和)/2开始往下找这个值能否由这些硬币凑出。要注意的是,可以由前n个硬币组成那样也是可以组成的面值。


代码:

#include <cstdio>
#include <cstring>

const int N = 105;
const int maxn = N * 500;

int v[N];
bool dp[maxn];
int n, sum;

void init () {

	memset (dp, false, sizeof (dp));
	dp[0] = true;

	for (int i = 0; i < n; i++)
		for (int j = sum; j >= v[i]; j--) {
				
			if (dp[j - v[i]])  
				dp[j] = true;
		//	dp[j] = dp[j - v[i]]; 这样写的话就否定了仅仅前面的i- 1个硬币就组成j的情况。 
		}
}

int main () {

	int t;
	scanf ("%d", &t);
	while (t--) {

		scanf ("%d", &n);
		sum = 0;
		for (int i = 0; i < n; i++) {

			scanf ("%d", &v[i]);
			sum += v[i];
		}
		
		init ();

		int i;
		for (i = sum / 2; i >= 0; i--) 
			if (dp[i])
				break;

		printf ("%d\n", sum - 2 * i);
				
	}
	return 0;
}