【题目】
输入一个整数n,求1到n这n个整数的十进制表示中1出现的次数。比如输入12,1到12这些整数中包含1的数字有1,10,11,12,1一共出现了5次。
【思路】
参考链接https://www.cnblogs.com/lengender-12/p/6876897.html
规律如下:
- 若第i位(自右向左,从1开始标号)上的数字为0,则当前位置出现1的次数与高位有关(若无高位,则为0),出现次数为高位乘以当前位置的权重(10^(i-1))。
- 解释:将第i位的高位看成一部分,将第i位的低位看成一部分,则当前位置为1时,低位元素取值任意,每一位都可取[0,9]共10个数字,因此低位的总取值情况为10^(i-1);而高位(假设高位元素为high)作为一个整体取值情况为0-(high-1)(高位取high时超过了范围),共有k种情况,因此总次数为 高位元素值*10^(n-1)
- 举例:n=21305,当十位取值为1时,个位取值为[0-9],共有10种情况(10^1),高位取值范围为[0,212],因此共有213*10=2130种情况
- 若第i位上的数字为1,则当前位置出现1的次数既受高位影响也受低位影响(如果没有,视低位为0),出现次数为高位数*当前位置的权重+低位数+1
- 解释:同第一种情况类似,把第i位的高位值high和低位low看成两部分时,为了避免超过范围,当高位取值范围为[0, high-1]时,低位取值任意,出现次数为高位数*当前位置的权重;而当高位取值为high时,低位取值范围只能是[0,low],此时出现次数为低位数+1
- 举例:n=21305,当千位取1时分两种情况讨论:
- 高位取值为[0,1]时,低位任意取值,出现次数为2*10^3
- 高位取值为2时,低位取值范围为[0,305],出现次数为305+1
- 若第i位上的数字大于1,则当前位置出现1的次数仅与高位有关,出现次数为(高位数+1)*当前位置的权重
- 解释:当前位置元素大于1时,高位取值范围为[0,high],低位取值任意(因为第i位值大于1,因从结果不会超过范围)
- 举例:n=21305,百位取1时,高位取值范围为[0,2],低位取值任意,总共有10^3种取值情况
【C++代码】
class Solution
{
public:
int NumberOf1Between1AndN_Solution(int n)
{
if(n<1) return 0;
if(n<9) return 1;
int count=0, high=0, k=0, cur=0;
for(int i=1; k=n/i; i*=10)
{
high = k/10;
cur = k%10;
count += high*i;
if(cur==1)
count += n-k*i+1;
else if(cur>1)
count += i;
}
return count;
}
};