2020牛客暑期多校训练营(第六场)H.Harmony Pairs 数位DP

题目链接:https://ac.nowcoder.com/acm/contest/5671/H
题目大意:
s(x)=x在十进制下所有位的数字之和。s(123)=1+2+3=6
现在给你一个n。问:满足0<=x<=y<=n并且s(x)>s(y)的(x, y)的对数。
图片说明

思路:看出来是数位DP了。但是不知道两个数怎么搞。
图片说明

#include<bits/stdc++.h>
#define LL long long
using namespace std;
const int mod=1e9+7;

char s[105];
LL f[105][2005][2][2][2];
//pos初始值为1000,防止负数产生
LL DFS(int pos, int sum, int va, int vb, int limit){
    if(pos==0){
        return sum>1000;//s(x)-s(y)>0
    }
    if(f[pos][sum][va][vb][limit]!=-1){
        return f[pos][sum][va][vb][limit];
    }
    int na=va?(s[pos]-'0'):9;
    int nb=vb?(s[pos]-'0'):9;
    LL res=0;
    for(int i=0; i<=na; i++){//枚举这位x的值
        for(int j=0; j<=nb; j++){//枚举这位y的值
            if(!limit&&i>j){//如果之前没有x<y,这位x的值不能比y的值大
                continue;
            }
            res+=DFS(pos-1, sum+i-j, va&&i==na, vb&&j==nb, limit||j>i);
            res%=mod;
        }
    }
    return f[pos][sum][va][vb][limit]=res;
}

int main() {
    memset(f, -1, sizeof(f));
    scanf("%s", s+1);
    int n=strlen(s+1);
    int L=1, R=n;
    while(L<R){
        swap(s[L], s[R]);
        L++, R--;
    }
    printf("%lld\n", DFS(n, 1000, 1, 1, 0));

    return 0;
}
全部评论

相关推荐

评论
点赞
收藏
分享
牛客网
牛客企业服务