CodeForces 258B Little Elephant and Elections

标签(空格分隔): 数位DP 背包DP


算出有i个幸运数字的方案,然后枚举上限做背包DP

#include<cstdio>
#include<cstring>
#include<iostream>
using namespace std;
typedef long long ll;
ll read()
{
	ll x=0,f=1;char ch=getchar();
	while(ch<'0'||ch>'9'){if(ch=='-')f=-1;ch=getchar();}
	while(ch>='0'&&ch<='9'){x=x*10+ch-'0';ch=getchar();}
	return x*f;
}
const int mod=1e9+7;
int m,f[15][15],g[15],dit[15];
ll calc(int x)
{
	ll res=1;
	for(int i=0;i<6;i++)res=res*(x-i)%mod;
	return res;
}
void MOD(int &x,int y)
{
	x+=y;
	if(x>=mod)x-=mod;
}
int dp(int step,int r,int lim)
{
	if(step==0)return r==0;
	if(!lim&&~f[step][r])return f[step][r];
	int x=lim?dit[step]:9,res=0;
	for(int i=0;i<=x;i++)
	{
		if(i==4||i==7)
		{
			if(r==0)continue;
			MOD(res,dp(step-1,r-1,lim&&i==x));
			continue;
		}
		res+=dp(step-1,r,lim&&i==x);
	}
	if(!lim)f[step][r]=res;
	return res;
}
int dfs(int step,int r,ll mul)
{
	if(step==0)return mul;
	int res=0;
	for(int i=0;i<=r;i++)if(g[i])
	{
		g[i]--;
		MOD(res,dfs(step-1,r-i,mul*(g[i]+1)%mod));
		g[i]++;
	}
	return res;
}
int main()
{
	memset(f,-1,sizeof(f));
	m=read();
	int len=0;
	while(m)dit[++len]=m%10,m/=10;
	for(int i=0;i<=len;i++)g[i]=dp(len,i,1);
	g[0]--;
	int ans=0;
	for(int i=1;i<=len;i++)MOD(ans,(ll)g[i]*dfs(6,i-1,1)%mod);
	printf("%d
",ans);
	return 0;
}
原文地址:https://www.cnblogs.com/ljzalc1022/p/9018902.html