约 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);
}
};