寻找存在的第n个丑数

什么是丑数:我们把只包含质因子 2、3 和 5 的数称作丑数(Ugly Number)。

判断数x是不是丑数:所以如果想判断一个数是不是丑数,可以判断这个数x能不能分别整除2、3、5。如果可以整除就让这个数除以2(或3或5)。直到x不能再整除2、3和5。此时若被处理完成的x等于1。则x是丑数。否则就不是丑数。

寻找存在的第n个丑数:我们可以可以建立一个计数器count,初始化为0。然后从1开始向后遍历正整数x。若遍历到的x是丑数,则count++。若不为丑数则不对count处理。当count==n时。输出x的值即可得到存在的第n个丑数。

```

#include <iostream>
using namespace std;

class Solution 
{
public:
	int nthUglyNumber(int n)
	{
		if (n == 1)
			return 1;
		int k = 1;
		int num = 1;
		while (k != n)
		{
			num++;
			int x = num;
			while (x != 1)
			{
				while (x % 2 == 0)
					x /= 2;
				while (x % 3 == 0)
					x /= 3;
				while (x % 5 == 0)
					x /= 5;
				if (x == 1)
					k++;
				x = 1;
			}
		}
		return num;
	}

};

int main()
{
	Solution a;
	clock_t start, end;
	start = clock();
	printf("%d\n", a.nthUglyNumber(1691));
	end = clock();
	printf("时间为%f\n",difftime(end , start)/1000);
	return 0;
}
}

运行截图:

 

```

这种暴力解法很好理解也很好实现,但是时间复杂度太高,所以我们可以根据丑数的定义找到更好的方法。

我们知道丑数的定义是只包含质因子 2、3 和 5的数,而且质数相乘不会得到其它质数,所以所有的丑数都可以由1*2^{num2}*3^{num3}*5^{num5}得到(其中numx是非负的整数),这明显是一个单调递增的数组,处了1之外的丑数,我们还可以用2或3或5乘某一个比当前要求的丑数小的丑数表示。例如8,9,10,12,15一定是某一个丑数乘以2,3,5三个质数中的某一个得到。所以我们的思路就是:从第一个丑数开始,一个个数丑数,并确保数出来的丑数是递增的,直到数到第n个丑数,得到答案。所以如果我们可以找到正确的数丑数的方法就可以了。

我们可以建立一个长度为n的数组dp存放存在的丑数,建立三个指针num2,num3,num5指向dp[0]。已知dp[0]=1。我们从下标0开始在寻找下一个丑数时,找到2*dp[num2]、3*dp[num3]、5*dp[num5]中的最小值作为当前丑数,并且让numx++。当dp下标移动到n时,那么dp[n]就是存在的第n个丑数。

...

#include <iostream>
using namespace std;

class Solution 
{
public:
	int nthUglyNumber(int n) 
	{
		int *dp = new int[n];
		dp[0] = 1;
		if (n == 1)
		{
			return 1;
		}
		int nums2 = 0;
		int nums3 = 0;
		int nums5 = 0;
		for (int i = 1; i < n; i++)
		{
			dp[i] = fmin(dp[nums2] * 2, fmin(dp[nums3] * 3, dp[nums5] * 5));
			if (dp[i] == dp[nums2] * 2)
				nums2++;
			if (dp[i] == dp[nums3] * 3)
				nums3++;
			if (dp[i] == dp[nums5] * 5)
				nums5++;
		}
		return dp[n-1];
	}
};

int main()
{
	Solution a;
	printf("%d\n", a.nthUglyNumber(1691));
	return 0;
}

...

 

时间复杂度:O(n)。需要计算数组 dp 中的 n 个元素,每个元素的计算都可以在 O(1) 的时间内完成。

空间复杂度:O(n)。空间复杂度主要取决于数组dp 的大小。

  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值