跳转至

约 20 个字 67 行代码 预计阅读时间 1 分钟

数位dp

数位dp模板 力扣2376

// mask 可选 表示
class Solution {
public:
    int countSpecialNumbers(int n) {
        string s = to_string(n);
        int m = s.length();
        vector<vector<int>> memo(m, vector<int>(1 << 10, -1)); // -1 表示没有计算过
        auto dfs = [&](auto&& dfs, int i, int mask, bool is_limit, bool is_num) -> int {
            if (i == m) {
                return is_num; // is_num 为 true 表示得到了一个合法数字
            }
            if (!is_limit && is_num && memo[i][mask] != -1) {
                return memo[i][mask]; // 之前计算过
            }
            int res = 0;
            if (!is_num) { // 可以跳过当前数位
                res = dfs(dfs, i + 1, mask, false, false);
            }
            // 如果前面填的数字都和 n 的一样,那么这一位至多填数字 s[i](否则就超过 n 啦)
            int up = is_limit ? s[i] - '0' : 9;
            // 枚举要填入的数字 d
            // 如果前面没有填数字,则必须从 1 开始(因为不能有前导零)
            for (int d = is_num ? 0 : 1; d <= up; d++) {
                if ((mask >> d & 1) == 0) { // d 不在 mask 中,说明之前没有填过 d
                    res += dfs(dfs, i + 1, mask | (1 << d), is_limit && d == up, true);
                }
            }
            if (!is_limit && is_num) {
                memo[i][mask] = res; // 记忆化
            }
            return res;
        };
        return dfs(dfs, 0, 0, true, false);
    }
};

数位dp模板2 力扣233

class Solution {
public:
    int countDigitOne(int n) {
        string s = to_string(n);
        int m = s.length();
        vector<vector<int>> memo(m, vector<int>(m, -1)); // -1 表示没有计算过
        auto dfs = [&](auto&& dfs, int i, int cnt1, bool is_limit) -> int {
            if (i == m) {
                return cnt1;
            }
            if (!is_limit && memo[i][cnt1] >= 0) { // 之前计算过
                return memo[i][cnt1];
            }
            int res = 0;
            int up = is_limit ? s[i] - '0' : 9;
            for (int d = 0; d <= up; d++) { // 枚举要填入的数字 d
                res += dfs(dfs, i + 1, cnt1 + (d == 1), is_limit && d == up);
            }
            if (!is_limit) {
                memo[i][cnt1] = res;
            }
            return res;
        };
        return dfs(dfs, 0, 0, true);
    }
};

颜色主题调整

评论区~

有用的话请给我个赞和 star => GitHub stars
快来跟我聊天~