这个题目看似很简单,其实是一个复杂的过程。
(一)如果按照平常的思路,依次遍历1到N计算1出现的次数,这样做非常的简单,但是时间是一个苦恼的问题,运算量过大
按照平常思路的代码如下:
#include <iostream>
using namespace std;
int getCount(int num)
{
int count = 0;
while(num)
{
count+=(num%10==1?1:0);
num /= 10;
}
return count;
}
int main()
{
int num;
cout<<"请输入一个整数"<<endl;
cin>>num;
int count = 1;
for(int i=2; i<=num; i++)
{
count+=getCount(i);
}
cout<<count<<endl;
}
(二)大家看到这个题的时候,肯定会想到这个题会有更优的解法,我在看这个题目的时候由于没有静下心来思考,整理的逻辑非常的混乱,不过我相信,只要静下心来就可以找到规律。
假设有一个正整数:abcde我们只需要讨论在a,b,c,d,e各个位上出现1的次数即可,最终的结果就是把每次出现1的次数加和。
现在讨论c位上出现1的次数:分为三种情况
(1)c=0,例如abcde=12013,则c位为1的数字有100-199,1100-1199........11100-11199 一共是1200个c位为1的数,可见如果c=0,c位为1的次数为ab*(de的位数)
(2)c=1,例如abcde=12113,只看比第一种情况多了哪些带有1的数字,12100-12113多了14个数,可见如果c=1,c位为1的次数为ab*(de的位数)+de+1;
(3)c>1, 例如abcde = 12213 可知c位为1的数有(ab+1)*(de的位数)个
根据上述的规律,代码实现如下:
#include <iostream>
using namespace std;
int main()
{
int num;
cout<<"请输入一个正整数:"<<endl;
cin>>num;
int count = 0,temp=1;
int former,mid,latter;
while(num/temp != 0)
{
latter = num%temp;
former = num/(temp*10);
mid = (num/temp)%10;
switch(mid)
{
case 0:
count+=former*temp;
break;
case 1:
count+=former*temp+latter+1;
break;
default:
count+=(former+1)*temp;
}
temp *=10;
}
cout<<count<<endl;
}
上述方法的时间复杂度只与整数N的位数有关。与第一种方法相比大大节省了时间。