有多少个整数序列符合如下条件:
输出答案,对取模。
输出答案,对取模。
艹C题又要dp
考虑思想,令的前个元素中和为的总可能数(),则可得伪代码:
dp[0][0] = 1
for i = 0 to N-1 // 逐个位置考虑
for j = 0 to K-1 // 考虑所有和的情况,无需考虑K
for k = 1 to M // 1-M的每个选择
if j + k <= K: // 限制条件
dp[i + 1][j + k] += dp[i][j] // 更新dp[i+1]
时间复杂度为,可以通过。
其实还可以利用前缀和优化。
不难发现,
其中,具体的值请自行推导。
因此,我们可以记录的前缀和:
则。
因此,时间复杂度为。
强烈建议读者独立推导并实现该方法。 前缀和优化的算法在E、F题中很常见。
#include <cstdio>
#define MOD 998244353
#define maxn 200005
using namespace std;
inline void mod(int& x) { if(x >= MOD) x -= MOD; }
int dp[2][maxn];
int main()
{
int n, m, k;
scanf("%d%d%d", &n, &m, &k);
dp[0][0] = 1;
for(int i=0; i<n; i++)
{
int c = i & 1, p = c ^ 1;
for(int j=0; j<=k; j++)
dp[p][j] = 0;
for(int j=0; j<k; j++)
for(int d=1; d<=m && d<=k-j; d++)
mod(dp[p][j + d] += dp[c][j]);
}
int ans = 0;
for(int i=1; i<=k; i++)
mod(ans += dp[n & 1][i]);
printf("%d\n", ans);
return 0;
}
给定整数序列。
有个查询。每个查询的格式如下:
输出行,第行是第个查询的答案。
题目换句话说就是:求出现的位置中,在区间内的有多少个?
因此,我们很容易想到先预处理中每个数出现的位置,存入vector
,查询时二分即可。
注意二分边界。
#include <cstdio>
#include <vector>
#include <algorithm>
#define maxn 200005
using namespace std;
vector<int> pos[maxn];
int main()
{
int n, q;
scanf("%d", &n);
for(int i=0; i<n; i++)
{
int a; scanf("%d", &a);
pos[a].push_back(i);
}
scanf("%d", &q);
while(q--)
{
int l, r, x;
scanf("%d%d%d", &l, &r, &x);
printf("%d\n", int(
lower_bound(pos[x].begin(), pos[x].end(), r) -
lower_bound(pos[x].begin(), pos[x].end(), --l)
));
}
return 0;
}