C++实现统计从1到n中1出现的次数(另外一种方法)

这种方法不用每次都把数字截取为三段

Problem

 

Consider a function which, for a given whole number n, returns the number of ones required when writing out all numbers between 0 and n. 
For example, f(13)=6. Notice that f(1)=1. What is the next largest n such that f(n)=n?

 

1. 思考

 

1位数:0~9
       个位数为1:1,共1次
       故0~9之间,1的个数为1


2位数:10~99

       个位数为1:11, 21, 31, …, 91,共9次
       十位数为1:10, 11, 12, …, 19,共10次

       故0~99之间,1的个数为1+9+10=20次

 

       也可以这样考虑:
       在0~99之间个位有10个0~9,故有10*1=10次,而十位为1的有10次,共20次
       20=10*1+10

 

3位数:100~999
       个位数为1:101, 111, 121, …, 191  ——  10个
                             201, 211, 221, …, 291  ——  10个
                             …
                             901, 211, 221, …, 291  ——  10个
                             共9个10,是90次

 

       十位数为1:110, 111, 112, …, 119  ——  10个
                             210, 211, 212, …, 219  ——  10个
                             …
                             910, 911, 912, …, 919  ——  10个
                             共9个10,是90次

 

       百位数为1:100, 101, 102, …, 199  ——  100个
                             共100次

 

       故0~999之间,1的个数为20+90+90+100=300次

 

       也可以这样考虑:
       0~999之间:十位个位共有10个0~99,故有10*20=200次,而百位为1的有100次
       共200+100=300次
       300=10*20+100

 

4位数:1000~9999
       个位数为1:1001, 1011, 1021, …, 1091  ——  10个
                             1101, 1111, 1121, …, 1191  ——  10个
                             1201, 1211, 1221, …, 1291  ——  10个
                             …
                             9001, 9011, 9021, …, 9091  ——  10个
                             …
                             9801, 9811, 9821, …, 9891  ——  10个
                             9901, 9911, 9921, …, 9991  ——  10个
                             共90个10,是900次

 

       十位数为1:1010, 1011, 1012, …, 1019  ——  10个
                             1110, 1111, 1112, …, 1119  ——  10个
                             1210, 1211, 1212, …, 1219  ——  10个
                             …
                             9010, 9011, 9012, …, 9019  ——  10个
                             …
                             9810, 9811, 9812, …, 9819  ——  10个
                             9910, 9911, 9912, …, 9919  ——  10个
                             共90个10,是900次

 

       百位数为1:1100, 1101, 1102, …, 1199  ——  100个
                             2100, 2101, 2102, …, 2199  ——  100个
                             …
                             9100, 9101, 9102, …, 9199  ——  100个
                             共9个100,是900次

 

       千位数为1:1000, 1001, 1002, …, 1999  ——  1000个
                             共1000次

 

       故0~9999之间,1的个数为300+900*900*900+1000=4000

 

       也可以这样考虑:
       0~9999之间:百位十位个位共有10个0~999,故有10*300=3000次,而千位为1的有1000次
       共3000+1000=4000次
       4000=10*300+1000

 

2. 规律

 

       为方便观察,将这些数据列出,如下:

       0~9:1
       0~99:20=10*1+10
       0~999:300=10*20+100
       0~9999:4000=10*300+1000
       0~99999:50000=10*4000+10000
       0~999999:600000=10*50000+100000
       …

 

       从中,我们不难发现规律:

       a1=1
       a2=10*a1+10
       …
       am=10*am-1+10m-1
 

      即0~9…9(m个9)之间1的个数为am=10*am-1+10m-1个;

 

3. 实际计算

 

       找到这些规律并不难,可是对于任意给定的数n,怎样能快速得到0~n之间1的个数呢?

 

       例如,n=1234,计算步骤如下:
       (1) 将0~1234分为2个部分:0~999,1000~1234;
       (2) 有1个0~999,共300次;
       (3) 应该加上千位为1的次数,共234+1次,即n%1000+1=235次;
       (4) 此时,就剩下0~234了,不满1个0~999,故继续拆开:0~99,100~199,200~234;共有2个0~99,共2*20=40次;
       (5) 因百位数为2>1,故应该加上百位为1的次数,共100次;(从拆分的结果也很容易得出)
       (6) 此时,就剩下200~234了,实际上就是剩下0~34了,不满1个0~99,故继续拆开:0~9,10~19,20~29,30~34;共有3个0~9,共3*1=3次;
       (7) 因十位数为3>1,故应该加上十位为1的次数,共10次;
       (8) 此时,就剩下30~34了,实际上就是剩下0~4了,共1次(即因个位数为4>1,故应该加上个位为1的次数,共1次);

      

       故0~1234之间1的个数为300+235+40+100+3+10+1=689个

 

       如果,某个位的数字为1或者0,情况又是怎样呢?

 

       例如,n=20140,计算步骤如下:
       (1) 将0~20140分为3个部分:0~9999,10000~19999,20000~20140;
       (2) 有2个0~9999,共2*4000=8000次;
       (3) 应该加上万位为1的次数,共10000次,
       (4) 此时,就剩下0~140了,不满1个0~999,故拆为0~99,100~140;共有1个0~99,共1*20=20次;
       (5) 因百位为1=1,故应该加上百位为1的次数,共40+1次,即n%100+1=41次;
       (6) 此时,就剩下0~40了,共有4个0~9,共4*1=4次;
       (7) 因十位为4>1,故应该加上十位为1的次数,共10次;
       (8) 此时就剩下40了,实际上就是0,共0次;

 

       故0~20140之间1的个数为8000+10000+20+41+4+10=18075

 

4. 求解算法

 

-543210
a5000040003002010
x-1234Empty

 

       n=1234,0~n之间1的个数计算方法如下:

       count = 0

 

       count += x[4] * a[3] + 1000,             if x[4] > 1
       count += x[4] * a[3] + n%1000 + 1, if x[4] = 1

 

       count += x[3] * a[2] + 100,             if x[3] > 1
       count += x[3] * a[2] + n%100 + 1, if x[3] = 1

 

       count += x[2] * a[1] + 10,             if x[2] > 1
       count += x[2] * a[1] + n%10 + 1, if x[2] = 1

 

       count += x[1] * a[0] + 1,             if x[1] > 1
       count += x[1] * a[0] + n%1 + 1, if x[1] = 1

 

       即count += x[4] * a[3] + (x[4]>1) ? 1000 : ((x[4] =1) ? (n%1000 + 1) : 0)
           count += x[3] * a[2] + (x[3]>1) ? 100 : ((x[3] =1) ? (n%100 + 1) : 0)
           count += x[2] * a[1] + (x[2]>1) ? 10 : ((x[2] =1) ? (n%10 + 1) : 0)
           count += x[1] * a[0] + (x[1]>1) ? 1 : ((x[1] =1) ? (n%1 + 1) : 0)


  1. //计算正整数n中1的个数   
  2. long long CalculationTimes::GetTimes(long long n)  
  3. {  
  4.     long long count=0;  
  5.   
  6.     //0     :0   
  7.     //0~9   :1    = 10 * 0   + 1   
  8.     //0~99  :20   = 10 * 1   + 10   
  9.     //0~999 :300  = 10 * 20  + 100;   
  10.     //0~9999:4000 = 10 * 300 + 1000   
  11.     long long a=0;      //0,1,20,300,4000,...   
  12.     long long weight=1; //1,10,100,1000,...   
  13.       
  14.     int x;          //数的当前位   
  15.   
  16.   
  17.     long long temp=n;  
  18.     while(temp)  
  19.     {  
  20.         x=temp%10;  //求出当前的位
  21.   
  22.         count+=x*a; //每次统计的时候,当前位,乘以的a,实际就是去掉当前位其它位1出现的次数                  
  23.                     //从0到lengh-1位的9之间1   出现的次数
  24.  
  25.         //加上当前位为1的个数   
  26.         if(x>1)         //若该位>1,则当前位为1的个数为10^(m-1)   
  27.             count+=weight;  
  28.         else if(x==1)   //若该位=1,则当前位为1的个数为该位右边的数字组成的数+1   
  29.             count+=n%weight+1;  
  30.           
  31.         a=10*a+weight;  //an=10 * a(n-1) + 10^(m-1), m为位数   
  32.         weight*=10;  
  33.   
  34.         temp/=10;  
  35.     }  
  36.     return count;  
  37. }  

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值