Atcoder beginner contest 336 -- E -- Digit Sum Divisible --- 题解(数位dp)

本文介绍了一种解决给定整数n下小于等于n的好整数个数问题的方法,利用数位dp和记忆化搜索策略,避免了不必要的重复计算,提高了代码效率。
摘要由CSDN通过智能技术生成

 

目录

 

E -- Digit Sum Divisibl

题目大意:

思路解析:

代码实现:


E -- Digit Sum Divisibl

题目大意:

给你一个整数n,让你找出小于等于n的数中一共有多少个好整数,并输出好整数的个数。对好整数的个数定义为如果一个数能被他的数位之和整除,则称这个数为好整数。例如 12 能被 3 整除。

n<=10^14。

思路解析:

看到数位之和,应该优先想到数位dp,那么我们想到最大的数位之和应该为14*9=126。则我们可以枚举每个数位之和k,看0-n中间是否有数位之和加起来刚好是k,并且刚好能被k整除。所以我们遍历次数为 126 * 10^14,这是不能接受的,但是对于每一个k来说,其实当中有许多重复遍历情况,所以可以用记忆化搜索来加速遍历行为。

代码实现:

import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.io.StreamTokenizer;
import java.util.*;

public class Main {
    static int[] dig = new int[15];
    static long[][][] dp;
    static int p;
    public static void main(String[] args) throws IOException {
         Scanner input = new Scanner(System.in);
         long n = input.nextLong();
         dp = new long[15][127][127];
         p = 0;
         while (n > 0){
             dig[p++] =(int) (n % 10);
             n /= 10;
         }
         long res = 0;
        for (int k = 1; k <= 126; k++) {
            for (int i = 0; i < 15; i++) {
                for (int j = 0; j < 127; j++) {
                    Arrays.fill(dp[i][j], -1);
                }
            }
            res += dfs(p - 1, 0, 0, k, false);
        }
        System.out.println(res);
    }
    // lim 表示当前位是否有限制,如果lim == false,表示当前位选择没有限制,可以选择 0-9,否则只能选择0-dig[u]
    // u 表示当前抉择到哪一位了‘
    // s 表示抉择到当前位的数位之和为多少,
    // t表示当前数模上k后的余数是多少
    public static long dfs(int u, int s, int t, int k, boolean lim){
        if (u == -1 && t == 0 && s == k) return 1;
        if (u == -1) return 0;
        if (lim && dp[u][s][t] != -1) return dp[u][s][t];
        int x = lim ? 9 : dig[u];
        long ans = 0;
        for (int i = 0; i <= x; i++) {
            ans += dfs(u - 1, s+i, (t * 10 + i) % k, k, i != x || lim);
        }
        if (lim) dp[u][s][t] = ans;
        return ans;
    }
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Studying~

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值