剑指Offer——整数1出现的次数

题目:求出1~13的整数中1出现的次数,并算出100~1300的整数中1出现的次数?为此他特别数了一下1~13中包含1的数字有1、10、11、12、13因此共出现6次,但是对于后面问题他就没辙了。ACMer希望你们帮帮他,并把问题更加普遍化,可以很快的求出任意非负整数区间中1出现的次数(从1 到 n 中1出现的次数)。

方法一:累加1~n中每个整数1出现的次数。通过对n求余判断整数的个位数是不是1,如果这个数字大于10,则除以10之后在判断个位数字是不是1。

方法二:f(n)的意思是1~n中这n个整数十进制出现1的次数,将n拆分成两个部分,最高的一位high和其他位last。

1.例如:n=1234,high=1,pow=1000,last=234

      (1)1~999这个范围1的个数是f(pow-1)
      (2)1000~1234这个范围1的个数需要分为两部分:
                 a.千分位是1的个数:千分位为1的个数刚好就是234+1(last+1),注意,这儿只看千分位,不看其他位
                 b.其他位是1的个数:即是234中出现1的个数,为f(last)
       所以全部加起来是f(pow-1) + last + 1 + f(last);

2.例子如3234,high=3, pow=1000, last=234

可以将数字范围分成两部分1~999,1000~1999,2000~2999和3000~3234

  (1)1~999这个范围1的个数是f(pow-1)
         (2)1000~1999这个范围1的个数需要分为两部分:
                  a.千分位是1的个数:千分位为1的个数刚好就是pow,注意,这儿只看千分位,不看其他位
                  b.其他位是1的个数:即是999中出现1的个数,为f(pow-1)
       (3)2000~2999这个范围1的个数是f(pow-1)
      (4)3000~3234这个范围1的个数是f(last)
    所以全部加起来是pow + high*f(pow-1) + f(last);

public class Solution {
    public int NumberOf1Between1AndN_Solution(int n) {
       return f(n);
    }
    private int f(int n){
         if(n<=0){
            return 0;
        }
        String s = String.valueOf(n);
        int high=s.charAt(0)-'0';
        int pow=(int)Math.pow(10,s.length()-1);
        int last=n-high*pow;
        if(high==1){
            return f(pow-1)+1+f(last)+last;
        }
        else{
            return pow+high*f(pow-1)+f(last);
        }
    }
}

 

 

你可能感兴趣的:(LeetCode刷题)