google面试题及我的算法(2)——0~n之间1的个数(完美版)

本博客(http://blog.csdn.net/livelylittlefish)贴出作者(三二一、小鱼)相关研究、学习内容所做的笔记,欢迎广大朋友指正!


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. 求解算法

 

- 5 4 3 2 1 0
a 50000 4000 300 20 1 0
x - 1 2 3 4 Empty

 

       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)

 

5.源程序

/************************************************************************ * 0~n之间1的个数,如f(13)=6 * 1,2,3,4,5,6,7,8,9,10,11,12,13.1的个数为6 * 要求:输入一个正整数n,求出f(n),及求解f(n)=n ************************************************************************/ #include #include #include class CalculationTimes { public: CalculationTimes(){} ~CalculationTimes(){} long long GetTimes(long long n); }; //计算正整数n中1的个数 long long CalculationTimes::GetTimes(long long n) { long long count=0; //0 :0 //0~9 :1 = 10 * 0 + 1 //0~99 :20 = 10 * 1 + 10 //0~999 :300 = 10 * 20 + 100; //0~9999:4000 = 10 * 300 + 1000 long long a=0; //0,1,20,300,4000,... long long weight=1; //1,10,100,1000,... int x; //数的当前位 long long temp=n; while(temp) { x=temp%10; count+=x*a; //加上当前位为1的个数 if(x>1) //若该位>1,则当前位为1的个数为10^(m-1) count+=weight; else if(x==1) //若该位=1,则当前位为1的个数为该位右边的数字组成的数+1 count+=n%weight+1; a=10*a+weight; //an=10 * a(n-1) + 10^(m-1), m为位数 weight*=10; temp/=10; } return count; } //显示菜单 void show_menu() { printf("---------------------------------------------/n"); printf("input command to test the program/n"); printf(" i or I : input n to test/n"); printf(" g or G : get n to enable f(n)=n/n"); printf(" q or Q : quit/n"); printf("---------------------------------------------/n"); printf("$ input command >"); } void main() { char sinput[10]; long long n; show_menu(); scanf("%s",sinput); while(stricmp(sinput,"q")!=0) { int t=0; long long count=0; if(stricmp(sinput,"i")==0) { printf(" please input an integer:"); scanf("%lld",&n); count=0; CalculationTimes obj; t=GetTickCount(); count=obj.GetTimes(n); t=GetTickCount()-t; printf("/n count=%lld time=%d/n/n",count,t); } else if(stricmp(sinput,"g")==0) { CalculationTimes obj; n=2; t=GetTickCount(); while(1) { count=obj.GetTimes(n); if(count==n) break; n++; } t=GetTickCount()-t; printf("/n f(n)=n=%lld time=%d/n/n",n,t); } //输入命令 printf("$ input command >"); scanf("%s",sinput); } }  

6. 运行结果

 

google面试题及我的算法(2)——0~n之间1的个数(完美版)_第1张图片

 

由算法本身和程序运行结果来看,本算法的效率非常高,将问题的解法转化为纯粹的数学计算,只需要简单的计算即可得到结果。

你可能感兴趣的:(算法研究)