给定一个十进制的正整数N,计算从1开始到N的的所有整数中1出现的个数

这个题目看似很简单,其实是一个复杂的过程。

(一)如果按照平常的思路,依次遍历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的位数有关。与第一种方法相比大大节省了时间。


 

 

  • 4
    点赞
  • 15
    收藏
    觉得还不错? 一键收藏
  • 3
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值