题意:T个家族的蚂蚁一共有A只,每个家族的蚂蚁编号分别为1,2...T,这些蚂蚁中的若干个组成一群去觅食,问这一群蚂蚁的数量在s到b之间的方案数一共有多少。
分析:设f[i][j]为前i种蚂蚁一共j只组成的方案数。则f[i][j]=sigma(f[i-1][j-0],f[i-1][j-1]...f[i-1][j-g[i]])。g[i]指编号为i种类的蚂蚁的数量。滚动数组以节约空间。
Code:
#include <algorithm> #include <iostream> #include <cstring> #include <string> #include <cstdio> #include <vector> #include <queue> #include <cmath> #include <map> #include <set> #define eps 1e-7 #define LL long long #define pb push_back #define Max(a,b) ((a)>(b)?(a):(b)) #define Min(a,b) ((a)<(b)?(a):(b)) using namespace std; const int MOD=1000000; const int inf=0x3f3f3f3f; const int maxn=1005; int sum[2][maxn*10]; int f[maxn]; int t,s,b,a; int main() { int x; scanf("%d %d %d %d",&t,&a,&s,&b); memset(f,0,sizeof(f)); for(int i=0;i<a;i++){ scanf("%d",&x); f[x]++; } memset(sum,0,sizeof(sum)); for(int i=0;i<=f[1];i++){ sum[0][i]=1; } int cur=1,pre=0; for(int i=2;i<=t;i++){ for(int j=0;j<=a;j++){ for(int k=0;k<=f[i];k++){ if(k>j) break; sum[cur][j]=(sum[cur][j]+sum[pre][j-k])%MOD; } } cur^=1;pre^=1; memset(sum[cur],0,sizeof(sum[cur])); } cur^=1; int ans=0; for(int i=s;i<=b;i++) ans=(ans+sum[cur][i])%MOD; printf("%d\n",ans); return 0; }