剑指offer-整数中1出现的次数

【题目】

输入一个整数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;
    }
};

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值