子序列为k的倍数的最大长度
链接:https://ac.nowcoder.com/acm/problem/15613
来源:牛客网
给一个数组 a,长度为 n,若某个子序列中的和为 K 的倍数,那么这个序列被称为“K 序列”。现在要你 对数组 a 求出最长的子序列的长度,满足这个序列是 K 序列。
解法1:
01背包要或者不要
dp[j]表示余数为j的最大长度,我们可以把所有的数转换为k的余数,不影响结果,所有数的范围就变小了,遍历0->k-1,表示当前余数为j,内循环给的序列,判断要或者不要, dp[(j+a[i])%k]= max((dp [(j+a[i]) %k],dp[j]+1);这里呢,这一轮循环与上一轮循环的前面和后面都有关,所以要用两个数组存,也可以用滚动数组。
#include <iostream>
#include <stdio.h>
using namespace std;
int main()
{
int n,k;
cin>>n>>k;
int a[100010];
for(int i=1;i<=n;i++)
{
scanf("%d",&a[i]);
a[i]=a[i]%k;
}
int dp[2][100010];
dp[0][0]=0;
for(int i=1;i<k;i++)
dp[0][i]=-0x3f3f3f;
int cur=0;
dp[0][a[1]]=1;
for(int i=2;i<=n;i++)
{
for(int j=0;j<k;j++)
{
dp[1-cur][(j+a[i])%k]=max((dp[cur][(j+a[i])%k]),dp[cur][j]+1);
}
cur=1-cur;
}
cout<<dp[cur][0]<<endl;
return 0;
}
解法2:还没看懂emm