poj 1338 ugly numbers和poj 2247Humble Numbers 解题报告(附详细分析)

poj 1338 ugly numbers

题目链接:http://poj.org/problem?id=1338

最直观的思路是验证每个数是否是ugly number,但是很费时间,效率低。

于是,我们需要从另一个角度思考,把1500个ugly number存在一个数组里。需要时直接调用。

思路:ugly number都是由2,3或5相乘得到,(除了1)

1

2=2*1

3=3*1

4=2*2

5=5*1

6=2*3//3*2

8=4*2

9=3*3

10=5*2

12=3*4

观察可以发现,从2开始,2*i, 3*j, 5*k,(i, j, k = 1.2.3.....)是按从小到大的顺序排列而成的。

所以可以利用一个算法,即比较2*i, 3*j, 5*k的大小,把最小的那个放进数组里,然后将对应的i, j, 或k加1;重复n次,就能得到所有的ugly number。

模拟算法实现:

ugly[1]=1

一,2*1, 3*1, 5*1

ugly[2] = 2*1;

二,2*2,3*1, 5*1

ugly[3] = 3*1;

三,2*2, 3*2,  5*1;

ugly[4] = 2*2;

..........

代码:

ps:将i, j, k分别设置为i_1, i_2, i_3;

#include <iostream>
#include <algorithm>
#include <stdio.h>
using namespace std;
#define N 1500		
int main()
{
	long ugly[N+1];
	int n;
	ugly[1] = 1;						//设置第一个为1
	int		i_1 = 1,
			i_2 = 1,
			i_3 = 1;
	for(int i = 2; i <= N; i++)			//设置循环次数,可以得到N个ugly number
		{
			ugly[i] = min(ugly[i_1]*2, min(ugly[i_2]*3, ugly[i_3]*5));		//比较三个数的大小,并将最小值放进数组
			if(ugly[i] == ugly[i_1]*2)				//将最小值对应的i_1, i_2,或i_3加1
				i_1++;
			if(ugly[i] == ugly[i_2]*3)
				i_2++;
			if(ugly[i] == ugly[i_3]*5)
				i_3++;
		}
	while (true)
	{
		scanf("%d", &n);
		if(n == 0)	break;
		printf("%d\n", ugly[n]);
	}
	return 0;
}


poj 2247Humble Numbers 

题目链接:http://poj.org/problem?id=2247

这道题的思路和上面完全一样,唯一变化的是输出格式的要求。

特别注意的是要用到英文知识:

一般来说个位数是1为st,个位数为2是nd,个位数为3是rd;但是以11, 12 ,13结尾的均为th,如111,112, 113.

#include <iostream>
#include <algorithm>
#include <stdio.h>
using namespace std;
#define N 5842		
int main()
{
	long humble[N+1];
	int n;
	humble[1] = 1;						
	int		i_1 = 1,
			i_2 = 1,
			i_3 = 1,
			i_4 = 1;
	for(int i = 2; i <= N; i++)			
		{
			humble[i] = min(humble[i_1]*2, min(humble[i_2]*3, min(humble[i_3]*5, humble[i_4]*7)));		
			if(humble[i] == humble[i_1]*2)				
				i_1++;
			if(humble[i] == humble[i_2]*3)
				i_2++;
			if(humble[i] == humble[i_3]*5)
				i_3++;
			if(humble[i] == humble[i_4]*7)
				i_4++;
		}
	while(scanf("%d",&n)&& n)
    {
        if(n%100==11)
        printf("The %dth humble number is %d.\n",n,humble[n]);
        else if(n%100==12)
        printf("The %dth humble number is %d.\n",n,humble[n]);
        else if(n%100==13)
        printf("The %dth humble number is %d.\n",n,humble[n]);
        else if(n%10==1)
        printf("The %dst humble number is %d.\n",n,humble[n]);
        else if(n%10==2)
        printf("The %dnd humble number is %d.\n",n,humble[n]);
        else if(n%10==3)
        printf("The %drd humble number is %d.\n",n,humble[n]);
        else
        printf("The %dth humble number is %d.\n",n,humble[n]);
    }
	return 0;
}






评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值