剑指43:整数1到n中1出现的次数

方法1:字符串方式

从最高位开始

class Solution {
public:
    int NumberOf1Between1AndN_Solution(int n)
    {
        //递归过程 次高位和最高位作为递归分界条件 
        //如21342 第一次计算1343到21342之间 最高位万位出现1的次数 和 其他位出现1的次数 递归计算1324
        //结束条件1 个位时
        if(n<=0)
            return 0;
        if(n<10)
            return 1;
        //每次用stringstream将int转成string 
        stringstream temp;
        temp<<n;
        string str;
        temp>>str;
        int result = 0,highcount,lowcount;
        //以1324为例
        //记录次高位到最低位的int型数值    next=324
        int next = n - (str[0]-'0')*pow(10,str.size()-1);  
        //计算最高位上1出现的次数   
        if(str[0]>'1')   //如 21324 10000到19999
            highcount = pow(10,str.size()-1);
        else
            highcount = next + 1;      
        //计算当前情况下,从next~n中除去最高位外其他位中1出现的次数
        lowcount = (str[0]-'0')*pow(10,str.size()-2)*(str.size()-1);
        //递归 下一位到最低位的int型 并加和
        result = highcount + lowcount + NumberOf1Between1AndN_Solution(next);
        return result;
    }
};

方法2:数学方法统计和计算

从最低位开始 找出当前数量级如百位为1时数的总和

 通过取余和除法 获得这一位左侧、当前位内容、右侧    加和统计

  //主要思路:设定整数点(如1、10、100等等)作为位置点i(对应n的各位、十位、百位等等),分别对每个数位上有多少包含1的点进行分析
    //根据设定的整数位置,对n进行分割,分为两部分,高位n/i,低位n%i
    //当i表示百位,且百位对应的数>=2,如n=31456,i=100,则a=314,b=56,此时百位为1的次数有a/10+1=32(最高两位0~31),每一次都包含100个连续的点,即共有(a%10+1)*100个点的百位为1
    //当i表示百位,且百位对应的数为1,如n=31156,i=100,则a=311,b=56,此时百位对应的就是1,则共有a%10(最高两位0-30)次是包含100个连续点,当最高两位为31(即a=311),本次只对应局部点00~56,共b+1次,所有点加起来共有(a%10*100)+(b+1),这些点百位对应为1
    //当i表示百位,且百位对应的数为0,如n=31056,i=100,则a=310,b=56,此时百位为1的次数有a/10=31(最高两位0~30)
    //综合以上三种情况,当百位对应0或>=2时,有(a+8)/10次包含所有100个点,还有当百位为1(a%10==1),需要增加局部点b+1
    //之所以补8,是因为当百位为0,则a/10==(a+8)/10,当百位>=2,补8会产生进位位,效果等同于(a/10+1)


class Solution {
public:
    int NumberOf1Between1AndN_Solution(int n)
    {
        int result=0;
        long long i=1;
        for(i;i<=n;i*=10){
            int a = n/i,b=n%i;
            result += (a + 8)/10*i +(a%10==1)*(b+1);  
        }
        return result;
    }
};

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值