吐槽一下,这题内存卡的比较紧
我#define int long long + 数组开了15150150*150 就爆内存了,一直以为我递归错误,查了好久。。。

用sum表示各个位之和,d表示枚举1到129的一个变量(因为sum要递归完了才能算出来,我们不知道sum究竟是多少,所以要枚举一下),mod表示后几位取模d以后得到的一个数。
那么只要递归到最深处(整个数字都算出来),d==sum(枚举的这个d刚好是这个数各个位之和),并且mod%d==0(整个数对各个位之和取模为0,就是题目中要求的),就符合条件进行返回。
然后用一个f数组(这个因为开了long long,要计算一下记忆化的有效数组大小),算出来是12
9=108,那么开110就好了,开大了就爆内存了。
然后只要满足“后面的数可以任意填”这个条件,就可以进行记忆,下次遇到相同的位置,就不必递归到最深处了。

这个题给我一个启发,当有些条件不清楚的时候,而数据范围给的又恰到好处,让你的枚举不会超时,那么就可以通过枚举的方法,来解决我们不知道某个变量的困扰。

#include <bits/stdc++.h>
using namespace std;
#define int long long
int f[15][110][110][110];
int a[15];
int dp(int pos,int d,int sum,int mod,bool flag)
{
    if(pos==0) return (d==sum&&mod%sum==0);
    if(flag&&f[pos][d][sum][mod]!=-1) return f[pos][d][sum][mod];
    int x=flag?9:a[pos];
    int ans=0;
    for(int i=0;i<=x;i++)
    {
        ans+=dp(pos-1,d,sum+i,(mod*10+i)%d,flag||i<x);
    }
    if(flag) f[pos][d][sum][mod]=ans;
    return ans;
}
int cul(int x)
{
    int pos=0;
    while(x)
    {
        a[++pos]=x%10;
        x/=10;
    }
    int res=0;
    for(int i=1;i<=12*9;i++)
        res+=dp(pos,i,0,0,false);
    return res;
}
signed main()
{
    int t;
    scanf("%lld",&t);
    int kase=0;
    memset(f,-1,sizeof(f));
    while(t--)
    {
        int n;
        scanf("%lld",&n);
        printf("Case %lld: %lld\n",++kase,cul(n));
    }
}