902.最大为 N 的数字组合
【LetMeFly】902.最大为 N 的数字组合「抽象出了函数,看着较为明白的代码 + 手推」
力扣题目链接:https://leetcode.cn/problems/numbers-at-most-n-given-digit-set/
给定一个按 非递减顺序 排列的数字数组 digits
。你可以用任意次数 digits[i]
来写的数字。例如,如果 digits = ['1','3','5']
,我们可以写数字,如 '13'
, '551'
, 和 '1351315'
。
返回 可以生成的小于或等于给定整数 n
的正整数的个数 。
示例 1:
输入:digits = ["1","3","5","7"], n = 100 输出:20 解释: 可写出的 20 个数字是: 1, 3, 5, 7, 11, 13, 15, 17, 31, 33, 35, 37, 51, 53, 55, 57, 71, 73, 75, 77.
示例 2:
输入:digits = ["1","4","9"], n = 1000000000 输出:29523 解释: 我们可以写 3 个一位数字,9 个两位数字,27 个三位数字, 81 个四位数字,243 个五位数字,729 个六位数字, 2187 个七位数字,6561 个八位数字和 19683 个九位数字。 总共,可以使用D中的数字写出 29523 个整数。
示例 3:
输入:digits = ["7"], n = 8 输出:1
提示:
1 <= digits.length <= 9
digits[i].length == 1
digits[i]
是从'1'
到'9'
的数digits
中的所有值都 不同digits
按 非递减顺序 排列1 <= n <= 109
方法一:排列组合 + 动态规划
有两种数字小于$n$:
- 数字位数直接小于$n$的
- 数字位数和$n$相同,但仍然小于$n$
长度短的数字
对于第一种情况,假设$n=1024$,那么所有的三位数都小于$n$
假设候选数字$digits = {2, 5}$(有$2$个),那么:
- 个位数有$2^1=2$个
- 两位数有$2^2=4$个
- 三位数有$2^3=8$个
所有的三位数有$2+4+8=14$个
长度和$n$相等的数字
假设$n=631$,$digits = {2, 6, 7}$
怎么计算长度为$3$的数字中,小于$n$的有多少个呢?
这里可以借助动态规划的思想,用两个变量$lessThan$和$equal$,分别代表遍历到$631$的某一位(记为$i$)时,“小于”和“等于”$631$前$i$位的$i$位数的个数。
说人话就是:假如当前遍历到了$631$的第$2$位(第一个数是$6$,第二个数是$3$)
那么$lessThan$就是小于$63$的两位数,$equal$就是等于$63$的两位数。
最终遍历完$631$的每一位后,$lessThan + equal$即为_小于等于_$631$的三位数
- 首先看$631$的前$1$位$6$:
- 小于$6$的$1$位数有一个($2$),因此$lessThan = 1$
- 等于$6$的$1$位数有一个($6$),因此$equal = 1$
- 接着看$631$的前$2$位$63$:
- 小于$63$的$2$位数有$4$个($lessThan = lessThan \times len(digits) + equal * lessThanThisWei = 1 \times 3 + 1\times 1 = 4$,小于$63$的包括
第一位就小于6,这一位任意
和第一位等于6,这一位必须小于3
,而小于$3$的数有$1$个),因此$lessThan = 4$ - 等于$63$的$2$位数有$0$个($equal = equal\times equalThisWei = 1\ times 0 = 0$,等于$63$的方案数为$第一位等于6的方案数\times这一位等于3的方案数$),因此$equal = 0$
- 小于$63$的$2$位数有$4$个($lessThan = lessThan \times len(digits) + equal * lessThanThisWei = 1 \times 3 + 1\times 1 = 4$,小于$63$的包括
- 最后看$631$的前$3$位$631$:
- 小于$631$的$3$位数有$12$个($lessThan = lessThan \times len(digits) + equal * lessThanThisWei = 4 \times 3 + 0\times 0 = 12$,而小于$1$的数有$0$个),因此$lessThan = 12$
- 等于$631$的$3$位数有$0$个($equal = equal\times equalThisWei = 0\ times 0 = 0$),因此$equal = 0$
因此小于等于$631$的三位数有$lessThan + equal = 12 + 0 = 12$个
(加上一位数$3$个和两位数$3\times3=9$个,由[2, 6, 7]
组成的小于等于$631$的数一共有$12+(3+9)=24$个)
- 时间复杂度$O((\log_{10}n)\times(\log_{10}n + len(digits)))$。前面求“短数字”的时间复杂度是$O((\log_{10}n)^2)$,后面求“等长数字”的时间复杂度是$O(\log_{10}n\times len(digits))$(这里题目中说$digits$是升序的,因此还可以实用二分查找,但是数据量不大,因此不是很有必要)
- 空间复杂度$O(1)$
AC代码
C++
1 |
|
同步发文于CSDN,原创不易,转载请附上原文链接哦~
Tisfy:https://letmefly.blog.csdn.net/article/details/127385999