AtCoder Beginner Contest 044 C (背包DP、思维、*1583)

C - Tak and Cards

题意:给定 N 个正整数 x1,x2,,xn 和一个正整数 A, 问从中选择 k 个数 (k>0), 满足:这 k 个数的平均值是 A 的方案总数是多少 (1N,xi,A50) ?

思路:从 N 个元素中选择 k 个元素, 并且使得它们的平均值是 A,即:从 N 个元素中选择 k 个元素, 使得它们的和为 k×A
考虑 背包DP, fi,j,k 表示从前 i 个元素中选择 j 个元素, 并且它们的和等于 k 的方案数。
有 : fi,j,k=fi1,j,k+fi1,j1,kxi。时间复杂度和空间复杂度均为 O(N2×2500)


点击查看代码
#include <bits/stdc++.h>

int main() {
	std::ios::sync_with_stdio(false);
	std::cin.tie(0); std::cout.tie(0);
	int n, a;
	std::cin >> n >> a;
	std::vector<int> x(n + 1);
	for (int i = 1; i <= n; i++) {
		std::cin >> x[i];
	}
	long long dp[n + 1][n + 1][2510];
	memset(dp, 0, sizeof(dp));
	dp[0][0][0] = 1;
	for (int i = 1; i <= n; i++) {
		for (int j = 0; j <= i; j++) {
			for (int k = 0; k < 2510; k++) {
				dp[i][j][k] += dp[i - 1][j][k];
				if (k - x[i] >= 0 && j >= 1) {
					dp[i][j][k] += dp[i - 1][j - 1][k - x[i]];
				}
			}
		}
	}
	long long res = 0;
	for (int i = 1; i <= n; i++) {
		res += dp[n][i][i * a];
	}
	std::cout << res << "\n";
	return 0;
}

接下来考虑空间优化 (滚动数组):

posted @   yanhy-orz  阅读(19)  评论(0编辑  收藏  举报
相关博文:
阅读排行:
· 地球OL攻略 —— 某应届生求职总结
· 周边上新:园子的第一款马克杯温暖上架
· Open-Sora 2.0 重磅开源!
· 提示词工程——AI应用必不可少的技术
· .NET周刊【3月第1期 2025-03-02】
点击右上角即可分享
微信分享提示