20701除法
难度级别: B; 编程语言:不限;运行时间限制:1000ms; 运行空间限制:51200KB; 代码长度限制:2000000B
试题描述
    输入正整数n,按从小到大的顺序输出所有满足表达式abcde/fghij=n的abcde和fghij,其中a~j恰好为数字0~9的一个排列。 如果没有符合题意的数,输出0。本题所说的五位数可以包括前导0的,如01234在这里也称为五位数。
输入
一个正整数n 
输出
若干行,每行包括两个符合要求的五位正整数(每行的两个数先大后小),两数之间用一个空格隔开。 
输入示例
62
输出示例
79546 01283
94736 01528
其他说明
数据范围 2≤n≤79;题目中的从小到大指的是输出数据每行的第一个数是从小到大的顺序。 

据题意可知,这道题可以用暴搜解决,不过一般的暴搜在这题上要每一位都用一个循环,这道题2个数,每个数5位(万位可以为0),按理说10位数,10个循环,对不?但这里只要枚举一个数,另一个数可以求出来的。我们当然选择枚举小的那个数(其实哪个都一样),也就是除数,那么被除数应该用这个枚举出来的数×n;如果枚举大的那个数,也就是被除数,那么除数应该用这个枚举出来的数÷n。我们用p[10]数组存储这两个数,同时方便全查找是否有重复。这是最基础的办法,也是最好理解的办法。

这题是刘汝佳一册P114或刘汝佳二册P182的题,第七讲“暴力求解法”的第一题,只是讲了一下思路,没给代码。在这之前说一下,这道题为什么一定要只要枚举一个数。我们可以知道,如果枚举0-9的所有排列,会有10!=3628800轮循环,听起来枚举量还不大,是吧?如果只枚举一个,循环量会掉到5!=120轮,比比程序运算量吧!

下面是代码,用的最笨的暴搜,这题n最大79,而且每种数据可能性不多,所以暴搜可以做。

本程序枚举的是除数,然后乘出被除数。

#include<iostream>
using namespace std;
int n,a,b,p[10],h,i;
bool isit,ih;//isit检查每组被除数和除数是否合法,ih检查最后是否有结果
int main()
{
cin>>n;
for(p[0]=0;p[0]<=9;p[0]++)//除数万位
for(p[1]=0;p[1]<=9;p[1]++)//除数千位
for(p[2]=0;p[2]<=9;p[2]++)//除数百位
for(p[3]=0;p[3]<=9;p[3]++)//除数十位
for(p[4]=0;p[4]<=9;p[4]++)//除数个位
{
if(p[0]!=p[1] && p[0]!=p[2] && p[0]!=p[3] && p[0]!=p[4] && p[1]!=p[2] && p[1]!=p[3] && p[1]!=p[4] && p[2]!=p[3] && p[2]!=p[4] && p[3]!=p[4])//查找除数是否有重复数字。其实这样可以省去很多后面的被除数运算量!如果算完除数有重复还算被除数,一看很多重复,会做很多无用功。
{
a=p[0]*10000+p[1]*1000+p[2]*100+p[3]*10+p[4];//计算除数
b=a*n;//算出被除数
if(9999<b && b<100000) //确保10000<=被除数<=99999,想一想,被除数为什么不能低于1w?
{
p[5]=b/10000;//以下是分离被除数
p[6]=(b%10000)/1000;
p[7]=(b%1000)/100;
p[8]=(b%100)/10;
p[9]=b%10;
isit=1;//看看这组数是不是无重复数字
for(h=0;h<=9;h++)
for(i=h+1;i<=9;i++)
if(p[h]==p[i]) isit=0;//被除数和除数全排查找有没有重复
if(isit)//如果是1就无重复
{
ih=1;//确保有结果
for(i=5;i<=9;i++) cout<<p[i];//切记先打被除数!忽略了一个事,这里可以直接输出b
cout<<' ';
for(i=0;i<=4;i++) cout<<p[i];//后打除数!
cout<<endl;
}
}
}
}
if(!ih) cout<<'0';//没结果出0
return 0;
}
04-29 02:15