丑数

http://acm.nefu.edu.cn/JudgeOnline/problemshow.php?problem_id=574

description

只有质数2,3,5,7这几个作为因子的数叫做,丑数,比如前20个丑数是(从小到大来说) 1,2,3,4,6,7,8,9,10,12,14,15,16,18,20,21,24,25和27.
							

input

我们给你个n(1<=m<=5842)当输入n为0结束。
							

output

输出第n个丑数。每个数一行。
							

sample_input

1
2
3
4
11

							

sample_output


1
2
3
4
12


关于丑数的含义在题目中已有解释,有的题目中忽略了“7”这个质因子,其实这都不是最重要的重要的是掌握其处理的方法。
首先,判断一个数是否为丑数的方法如下:
int find_uglynum(int a)
{
    while(a%2==0)//将这个数中的质因子 2 耗尽
       a/=2;
    while(a%3==0) 
       a/=3;
    while(a%5==0)
       a/=5;
    while(a/7==0)
       a/=7;
    if(a==1)
       return 1;
    else 
       return 0;
}


但是这种方法过于费时,下面给出该题解题思路:
         首先,第一个丑数为“1”,后面的每一个丑数都是有前一个丑数乘2、3、5或7而来,那么后一个丑数就是前一个乘这四个数得到的最小值,for example:第一个:1,第二个:1*2、1*3、1*5或1*7,显然为2,第三个:2*2,1*3,1*5或1*7,显然是3,第四个:2*2,,2*3,1*5,1*7为4,第五个:3*2,2*3,1*5,1*7……   聪明的你是否看明白了呢?
 
下面给出本题的代码:
#include <iostream>
#include <cstdio>
using namespace std;
#define min(a,b) ((a)<(b)?(a):(b))
#define min4(a,b,c,d) min(min(a,b),min(c,d))
int a[5850];
int main()
{
	//freopen("data.in","r",stdin);
	//freopen("data.out","w",stdout);

    int n=1;

    int p2,p3,p5,p7;
    p2=p3=p5=p7=1;
    a[1]=1;
    while(n<5843)//枚举5842个丑数,放在数组a里。
    {
        a[++n]=min4(2*a[p2],3*a[p3],5*a[p5],7*a[p7]);//从现在枚举的4个丑数里,先选择小的放在a里。
        if(a[n]==2*a[p2])p2++;//如果a[n]==2*a[p2],2*a[p2]可能是吧a[n]枚举出的数,这样p2++,也可能是重复的枚举,这样也是p2++,总之p2++。
        if(a[n]==3*a[p3])p3++;//同理。
        if(a[n]==5*a[p5])p5++;//同理。
        if(a[n]==7*a[p7])p7++;//同理。
    }
    while(scanf("%d",&n)&&n)
    {
        printf("%d\n",a[n]);//要谁找谁。
    }
    return 0;
}


你可能感兴趣的:(丑数)