comments | difficulty | edit_url |
---|---|---|
true |
中等 |
输入一个整数 n
,求1~n这n个整数的十进制表示中1出现的次数。
例如,输入12,1~12这些整数中包含1 的数字有1、10、11和12,1一共出现了5次。
示例 1:
输入:n = 12 输出:5
示例 2:
输入:n = 13 输出:6
限制:
1 <= n < 2^31
注意:本题与主站 233 题相同:https://leetcode.cn/problems/number-of-digit-one/
这道题实际上是求在给定区间
对于区间
不过对于本题而言,我们只需要求出区间
这里我们用记忆化搜索来实现数位 DP。从起点向下搜索,到最底层得到方案数,一层层向上返回答案并累加,最后从搜索起点得到最终的答案。
基本步骤如下:
- 将数字
$n$ 转为 int 数组$a$ ,其中$a[0]$ 为最低位,而$a[i]$ 为最高位; - 根据题目信息,设计函数
$dfs()$ ,对于本题,我们定义$dfs(pos, cnt, limit)$ ,其中:
-
pos
表示数字的位数,从末位或者第一位开始,一般根据题目的数字构造性质来选择顺序。对于本题,我们选择从高位开始,因此,pos
的初始值为len
; -
cnt
表示当前数字中包含的$1$ 的个数。 -
limit
表示可填的数字的限制,如果无限制,那么可以选择$[0,1,..9]$ ,否则,只能选择$[0,..a[pos]]$ 。如果limit
为true
且已经取到了能取到的最大值,那么下一个limit
同样为true
;如果limit
为true
但是还没有取到最大值,或者limit
为false
,那么下一个limit
为false
。
那么答案为
关于函数的实现细节,可以参考下面的代码。
时间复杂度
相似题目:
class Solution:
def countDigitOne(self, n: int) -> int:
@cache
def dfs(pos, cnt, limit):
if pos < 0:
return cnt
up = a[pos] if limit else 9
ans = 0
for i in range(up + 1):
ans += dfs(pos - 1, cnt + (i == 1), limit and i == up)
return ans
a = []
while n:
a.append(n % 10)
n //= 10
return dfs(len(a) - 1, 0, True)
class Solution {
private int[] a = new int[12];
private Integer[][] f = new Integer[12][12];
public int countDigitOne(int n) {
int i = -1;
for (; n > 0; n /= 10) {
a[++i] = n % 10;
}
return dfs(i, 0, true);
}
private int dfs(int pos, int cnt, boolean limit) {
if (pos < 0) {
return cnt;
}
if (!limit && f[pos][cnt] != null) {
return f[pos][cnt];
}
int up = limit ? a[pos] : 9;
int ans = 0;
for (int i = 0; i <= up; ++i) {
ans += dfs(pos - 1, cnt + (i == 1 ? 1 : 0), limit && i == up);
}
return f[pos][cnt] = ans;
}
}
class Solution {
public:
int countDigitOne(int n) {
int a[12]{};
int f[12][12];
memset(f, -1, sizeof f);
int i = -1;
for (; n; n /= 10) {
a[++i] = n % 10;
}
function<int(int, int, bool)> dfs = [&](int pos, int cnt, bool limit) -> int {
if (pos < 0) {
return cnt;
}
if (!limit && f[pos][cnt] != -1) {
return f[pos][cnt];
}
int up = limit ? a[pos] : 9;
int ans = 0;
for (int i = 0; i <= up; ++i) {
ans += dfs(pos - 1, cnt + (i == 1), limit && i == up);
}
return f[pos][cnt] = ans;
};
return dfs(i, 0, true);
}
};
func countDigitOne(n int) int {
a := [12]int{}
f := [12][12]int{}
for i := range f {
for j := range f[i] {
f[i][j] = -1
}
}
i := -1
for ; n > 0; n /= 10 {
i++
a[i] = n % 10
}
var dfs func(int, int, bool) int
dfs = func(pos, cnt int, limit bool) int {
if pos < 0 {
return cnt
}
if !limit && f[pos][cnt] != -1 {
return f[pos][cnt]
}
up := 9
if limit {
up = a[pos]
}
ans := 0
for i := 0; i <= up; i++ {
t := 0
if i == 1 {
t++
}
ans += dfs(pos-1, cnt+t, limit && i == up)
}
f[pos][cnt] = ans
return ans
}
return dfs(i, 0, true)
}
/**
* @param {number} n
* @return {number}
*/
var countDigitOne = function (n) {
let res = 0;
let i = 1;
while (i <= n) {
let high = ~~(n / i / 10);
let cur = ~~(n / i) % 10;
let low = n - ~~(n / i) * i;
switch (cur) {
case 0:
res += high * i;
break;
case 1:
res += high * i + low + 1;
break;
default:
res += (high + 1) * i;
}
i *= 10;
}
return res;
};
public class Solution {
public int CountDigitOne(int n) {
long mulk = 1;
int ans = 0;
for (int k = 0; n >= mulk; ++k) {
ans += (int) (n / (mulk * 10) * mulk) + (int) Math.Min(Math.Max(n % (mulk * 10) - mulk + 1, 0), mulk);
mulk *= 10;
}
return ans;
}
}
class Solution {
private var digits = [Int](repeating: 0, count: 12)
private var memo = [[Int?]](repeating: [Int?](repeating: nil, count: 12), count: 12)
func countDigitOne(_ n: Int) -> Int {
var n = n
var i = 0
while n > 0 {
digits[i] = n % 10
n /= 10
i += 1
}
return dfs(i - 1, 0, true)
}
private func dfs(_ pos: Int, _ count: Int, _ limit: Bool) -> Int {
if pos < 0 {
return count
}
if !limit && memo[pos][count] != nil {
return memo[pos][count]!
}
let upperLimit = limit ? digits[pos] : 9
var ans = 0
for i in 0...upperLimit {
ans += dfs(pos - 1, count + (i == 1 ? 1 : 0), limit && i == upperLimit)
}
if !limit {
memo[pos][count] = ans
}
return ans
}
}