最新C语言竞赛练习题40题答案.docx
《最新C语言竞赛练习题40题答案.docx》由会员分享,可在线阅读,更多相关《最新C语言竞赛练习题40题答案.docx(53页珍藏版)》请在冰豆网上搜索。
最新C语言竞赛练习题40题答案
C语言竞赛练习题1
1.求最大数
问555555的约数中最大的三位数是多少?
*问题分析与算法设计
根据约数的定义,对于一个整数N,除去1和它自身外,凡能整除N的数即为N的约数。
因此,最简单的方法是用2到N-1之间的所有数去除N,即可求出N的全部约数。
本题只要求取约数中最大的三位数,则其取值范围可限制在100到999之间。
*程序说明与注释
#include
voidmain()
{
longi;
intj;
printf("Pleaseinputnumber:
");
scanf("%ld",&i);
for(j=999;j>=100;j--)
if(i%j==0)
{
printf("Themaxfactorwith3digitsin%ldis:
%d,\\n",i,j);
break;
}
}
*运行结果
输入:
555555
输出:
Themaxfactorwith3digitsin555555is:
777
2.高次方数的尾数
求13的13次方的最后三位数
*问题分析与算法设计
解本题最直接的方法是:
将13累乘13次方截取最后三位即可。
但是由于计算机所能表示的整数范围有限,用这种“正确”的算法不可能得到正确的结果。
事实上,题目仅要求最后三位的值,完全没有必要求13的13次方的完整结果。
研究乘法的规律发现:
乘积的最后三位的值只与乘数和被乘数的后三位有关,与乘数和被乘数的高位无关。
利用这一规律,可以大大简化程序。
*程序说明与注释
#include
voidmain()
{
inti,x,y,last=1;/*变量last保存求X的Y次方过程中的部分乘积的后三位*/
printf("InputXandY(X**Y):
");
scanf("%d**%d",&x,&y);
for(i=1;i<=y;i++)/*X自乘Y次*/
last=last*x%1000;/*将last乘X后对1000取模,即求积的后三位*/
printf("Thelast3digitsof%d**%dis:
%d\\n",x,y,last%1000);/*打印结果*/
}
*运行结果
InputXandY(X**Y):
13**13
Thelast3digitsof13**13is:
253
InputXandY(X**Y):
13**20
Thelast3digitsof13**20is:
801
3.借书方案知多少
小明有五本新书,要借给A,B,C三位小朋友,若每人每次只能借一本,则可以有多少种不同的借法?
*问题分析与算法设计
本问题实际上是一个排列问题,即求从5个中取3个进行排列的方法的总数。
首先对五本书从1至5进行编号,然后使用穷举的方法。
假设三个人分别借这五本书中的一本,当三个人所借的书的编号都不相同时,就是满足题意的一种借阅方法。
*程序说明与注释
voidmain()
{
inta,b,c,count=0;
printf("TherearediffrentmethodsforXMtodistributebooksto3readers:
\\n");
for(a=1;a<=5;a++)/*穷举第一个人借5本书中的1本的全部情况*/
for(b=1;b<=5;b++)/*穷举第二个人借5本书中的一本的全部情况*/
for(c=1;a!
=b&&c<=5;c++)/*当前两个人借不同的书时,穷举第三个人借5本书
中的1本的全部情况*/
if(c!
=a&&c!
=b)/*判断第三人与前两个人借的书是否不同*/
printf(count%8?
"%2d:
%d,%d,%d":
"%2d:
%d,%d,%d\\n",++count,a,b,c);
/*打印可能的借阅方法*/
}
*运行结果
TherearediffrentmethodsforXMtodistributebooksto3readers:
1:
1,2,32:
1,2,43:
1,2,54:
1,3,25:
1,3,4
6:
1,3,57:
1,4,28:
1,4,39:
1,4,510:
1,5,2
11:
1,5,312:
1,5,413:
2,1,314:
2,1,415:
2,1,5
16:
2,3,117:
2,3,418:
2,3,519:
2,4,120:
2,4,3
21:
2,4,522:
2,5,123:
2,5,324:
2,5,425:
3,1,2
26:
3,1,427:
3,1,528:
3,2,129:
3,2,430:
3,2,5
31:
3,4,132:
3,4,233:
3,4,534:
3,5,135:
3,5,2
36:
3,5,437:
4,1,238:
4,1,339:
4,1,540:
4,2,1
41:
4,2,342:
4,2,543:
4,3,144:
4,3,245:
4,3,5
46:
4,5,147:
4,5,248:
4,5,349:
5,1,250:
5,1,3
51:
5,1,452:
5,2,153:
5,2,354:
5,2,455:
5,3,1
56:
5,3,257:
5,3,458:
5,4,159:
5,4,260:
5,4,3
4.数制转换
将任一整数转换为二进制形式
*问题分析与算法设计
将十进制整数转换为二进制的方法很多,这里介绍的实现方法利用了C语言能够对位进行操作的特点。
对于C语言来说,一个整数在计算机内就是以二进制的形式存储的,所以没有必要再将一个整数经过一系列的运算转换为二进制形式,只要将整数在内存中的二进制表示输出即可。
*程序说明与注释
#include
voidprintb(int,int);
voidmain()
{
intx;printf("Inputnumber:
");
scanf("%d",&x);
printf("numberofdecimalform:
%d\\n",x);
printf("it\'sbinaryform:
");
printb(x,sizeof(int)*8);/*x:
整数sizeof(int):
int型在内存中所占的字节数
sizeof(int)*8:
int型对应的位数*/
putchar(\'\\n\');
}
voidprintb(intx,intn)
{
if(n>0)
{
putchar(\'0\'+((unsigned)(x&(1<<(n-1)))>>(n-1)));/*输出第n位*/
printb(x,n-1);/*归调用,输出x的后n-1位*/
}
}
*运行结果
输入:
8
输出:
numberofdecimalform:
8
it\'sbunaryform:
0000000000001000
输入:
-8
输出:
numberofdecimalform:
-8
it\'sbinaryform:
111111*********0
输入:
32767
输出:
numberofdecimalform:
32767
it\'sbinaryform:
0111111111111111
输入:
-32768
输出:
numberofdecimalform:
-32768
it\'sbinaryform:
1000000000000000
输入:
128
输出:
numberofdecimalform:
128
it\'sbinaryform:
0000000010000000
5.打鱼还是晒网
中国有句俗语叫“三天打鱼两天晒网”。
某人从1990年1月1日起开始“三天打鱼两天晒网”,问这个人在以后的某一天中是“打鱼”还是“晒网”。
*问题分析与算法设计
根据题意可以将解题过程分为三步:
1)计算从1990年1月1日开始至指定日期共有多少天;
2)由于“打鱼”和“晒网”的周期为5天,所以将计算出的天数用5去除;
3)根据余数判断他是在“打鱼”还是在“晒网”;
若余数为1,2,3,则他是在“打鱼”
否则是在“晒网”
在这三步中,关键是第一步。
求从1990年1月1日至指定日期有多少天,要判断经历年份中是否有闰年,二月为29天,平年为28天。
闰年的方法可以用伪语句描述如下:
如果((年能被4除尽且不能被100除尽)或能被400除尽)
则该年是闰年;
否则不是闰年。
C语言中判断能否整除可以使用求余运算(即求模)
*程序与程序注释
#include
intdays(structdateday);
structdate{
intyear;
intmonth;
intday;
};
voidmain()
{
structdatetoday,term;
intyearday,year,day;
printf("Enteryear/month/day:
");
scanf("%d%d%d",&today.year,&today.month,&today.day);/*输入日期*/
term.month=12;/*设置变量的初始值:
月*/
term.day=31;/*设置变量的初始值:
日*/
for(yearday=0,year=1990;year{
term.year=year;
yearday+=days(term);/*计算从1990年至指定年的前一年共有多少天*/
}
yearday+=days(today);/*加上指定年中到指定日期的天数*/
day=yearday%5;/*求余数*/
if(day>0&&day<4)printf("hewasfishingatthatday.\\n");/*打印结果*/
elseprintf("Hewassleepingatthatday.\\n");
}
intdays(structdateday)
{
staticintday_tab[2][13]=
{{0,31,28,31,30,31,30,31,31,30,31,30,31,},/*平均每月的天数*/
{0,31,29,31,30,31,30,31,31,30,31,30,31,},
};
inti,lp;
lp=day.year%4==0&&day.year%100!
=0||day.year%400==0;
/*判定year为闰年还是平年,lp=0为平年,非0为闰年*/
for(i=1;iday.day+=day_tab[lp];
returnday.day;
}
*运行结果
Enteryear/month/day:
19911025
Hewasfishingatday.
Enteryear/month/day:
19921025
Hewassleepingatday.
Enteryear/month/day:
19931025
Hewassleepingatday
--
6.抓交通肇事犯
一辆卡车违反交通规则,撞人后逃跑。
现场有三人目击事件,但都没有记住车号,只记下车号的一些特征。
甲说:
牌照的前两位数字是相同的;乙说:
牌照的后两位数字是相同的,但与前两位不同;丙是数学家,他说:
四位的车号刚好是一个整数的平方。
请根据以上线索求出车号。
*问题分析与算法设计
按照题目的要求造出一个前两位数相同、后两位数相同且相互间又不同的整数,然后判断该整数是否是另一个整数的平方。
*程序与程序注释
#include
#include
voidmain()
{
inti,j,k,c;
for(i=1;i<=9;i++)/*i:
车号前二位的取值*/
for(j=0;j<=9;j++)/*j:
车号后二位的取值*/
if(i!
=j)/*判断二位数字是否相异*/
{
k=i*1000+i*100+j*10+j;/*计算出可能的整数*/
for(c=31;c*cif(c*c==k)printf("Lorry--No.is%d.\\n",k);/*若是,打印结果*/
}
}
*运行结果
Lorry_No.is7744
7.该存多少钱
假设银行一年整存零取的月息为0.63%。
现在某人手中有一笔钱,他打算在今后的五年中的年底取出1000元,到第五年时刚好取完,请算出他存钱时应存入多少。
*问题分析与算法设计
分析存钱和取钱的过程,可以采用倒推的方法。
若第五年年底连本带息要取1000元,则要先求出第五年年初银行存款的钱数:
第五年初存款=1000/(1+12*0.0063)
依次类推可以求出第四年、第三年......的年初银行存款的钱数:
第四年年初存款=(第五年年初存款+1000)/(1+12*0.0063)
第三年年初存款=(第四年年初存款+1000)/(1+12*0.0063)
第二年年初存款=(第三年年初存款+1000)/(1+12*0.0063)
第一年年初存款=(第二年年初存款+1000)/(1+12*0.0063)
通过以上过程就可以很容易地求出第一年年初要存入多少钱。
*程序与程序注释
#include
voidmain()
{
inti;
floattotal=0;
for(i=0;i<5;i++)/*i为年数,取值为0~4年*/
total=(total+1000)/(1+0.0063*12);/*累计算出年初存款数额,第五次的计算
结果即为题解*/
printf("Hemustsave%.2fatfirst.\\n",total);
}
*运行结果
Hemustsave4039.44atfirst
8.怎样存钱利最大
假设银行整存整取存款不同期限的月息利率分别为:
0.63%期限=1年
0.66%期限=2年
0.69%期限=3年
0.75%期限=5年
0.84%期限=8年
利息=本金*月息利率*12*存款年限。
现在某人手中有2000元钱,请通过计算选择一种存钱方案,使得钱存入银行20年后得到的利息最多(假定银行对超过存款期限的那一部分时间不付利息)。
*问题分析与算法
为了得到最多的利息,存入银行的钱应在到期时马上取出来,然后立刻将原来的本金和利息加起来再作为新的本金存入银行,这样不断地滚动直到满20年为止,由于存款的利率不同,所以不同的存款方法(年限)存20年得到的利息是不一样的。
分析题意,设2000元存20年,其中1年存i1次,2年存i2次,3年存i3次,5年存i5次,8年存i8次,则到期时存款人应得到的本利合计为:
2000*(1+rate1)i1*(1+rate2)i2*(1+rate3)i3*(1+rate5)i5*(1+rate8)i8
其中rateN为对应存款年限的利率。
根据题意还可得到以下限制条件:
0<=i8<=2
0<=i5<=(20-8*i8)/5
0<=i3<=(20-8*i8-5*i5)/3
0<=i2<=(20-8*i8-5*i5-3*i3)/2
0<=i1=20-8*i8-5*i5-3*i3-2*i2
可以用穷举法穷举所有的i8、i5、i3、i2和i1的组合,代入求本利的公式计算出最大值,就是最佳存款方案。
*程序与程序注释
#include
#include
voidmain()
{
inti8,i5,i3,i2,i1,n8,n5,n3,n2,n1;
floatmax=0,term;
for(i8=0;i8<3;i8++)/*穷举所有可能的存款方式*/
for(i5=0;i5<=(20-8*i8)/5;i5++)
for(i3=0;i3<=(20-8*i8-5*i5)/3;i3++)
for(i2=0;i2<=(20-8*i8-5*i5-3*i3)/2;i2++)
{
i1=20-8*i8-5*i5-3*i3-2*i2;
term=2000.0*pow((double)(1+0.0063*12),(double)i1)
*pow((double)(1+2*0.0063*12),(double)i2)
*pow((double)(1+3*0.0069*12),(double)i3)
*pow((double)(1+5*0.0075*12),(double)i5)
*pow((double)(1+8*0.0084*12),(double)i8);
/*计算到期时的本利合计*/
if(term>max)
{
max=term;n1=i1;n2=i2;n3=i3;n5=i5;n8=i8;
}
}
printf("Formaxinumprofit,heshouldsosavehismoneyinabank:
\\n");
printf("madefixeddepositfor8year:
%dtimes\\n",n8);
printf("madefixeddepositfor5year:
%dtimes\\n",n5);
printf("madefixeddepositfor3year:
%dtimes\\n",n3);
printf("madefixeddepositfor2year:
%dtimes\\n",n2);
printf("madefixeddepositfor1year:
%dtimes\\n",n1);
printf("Toal:
%.2f\\n",max);
/*输出存款方式*/
}
*运行结果
Formaxinumprofit,heshouldsosavehismoneyinabank:
madefixeddepositfor8year:
0times
madefixeddepositfor5year:
4times
madefixeddepositfor3year:
0times
madefixeddepositfor2year:
0times
madefixeddepositfor1year:
0times
Total:
8841.01
可见最佳的存款方案为连续四次存5年期。
9.捕鱼和分鱼
A、B、C、D、E五个人在某天夜里合伙去捕鱼,到第二天凌晨时都疲惫不堪,于是各自找地方睡觉。
日上三杆,A第一个醒来,他将鱼分为五份,把多余的一条鱼扔掉,拿走自己的一份。
B第二个醒来,也将鱼分为五份,把多余的一条鱼扔掉,保持走自己的一份。
C、D、E依次醒来,也按同样的方法拿走鱼。
问他们合伙至少捕了多少条鱼?
*问题分析与算法设计
根据题意,总计将所有的鱼进行了五次平均分配,每次分配时的策略是相同的,即扔掉一条鱼后剩下的鱼正好分成五份,然后拿走自己的一份,余下其它的四份。
假定鱼的总数为X,则X可以按照题目的要求进行五次分配:
X-1后可被5整除,余下的鱼为4*(X-1)、5。
若X满足上述要求,则X就是题目的解。
*程序与程序注释
#include
voidmain()
{
intn,i,x,flag=1;/*flag:
控制标记*/
for(n=6;flag;n++)/*采用试探的方法。
令试探值n逐步加大*/
{
for(x=n,i=1&&flag;i<=5;i++)
if((x-1)%5==0)x=4*(x-1)/5;
elseflag=0;/*若不能分配则置标记falg=0退出分配过程*/
if(flag)break;/*若分配过程正常结束则找到结果退出试探的过程*/
elseflag=1;/*否则继续试探下一个数*/
}
printf("Totalnumberoffishcatched=%d\\n",n);/*输出结果*/
}
*运行结果
Totalnumberoffishcatched=3121
*问题的进一步讨论
程序采用试探法,试探的初值为6,每次试探的步长为1。
这是过分保守的做法。
可以在进一步分析题目的基础上修改此值,增大试探的步长值,以减少试探次数。
*思考题
请使用其它的方法求解本题
10.出售金鱼
买卖提将养的一缸金鱼分五次出售系统上一次卖出全部的一半加二分之一条;第二次卖出余下的三分之一加三分之一条;第三次卖出余下的四分之一加四分之一条;第四次卖出余下的五分之一加五分之一条;最后卖出余下的11条。
问原来的鱼缸中共有几条金鱼?
*题目分析与算法设计
题目中所有的鱼是分五次出售的,每次卖出的策略相同;第j次卖剩下的(j+1)分之一再加1/(j+1)条。
第五次将第四次余下的11条全卖了。
假定第j次鱼的总数为X,则第j次留下:
x-(x+1)/(j+1)
当第四次出售完毕时,应该剩下11条。
若X满足上述要求,则X就是题目的解。
应当注意的是:
"(x+1)/(j+1)"应满足整除条件。
试探X的初值可以从23开始,试探的步长为2,因为X的值一定为奇数。
*程序说明与注释
#include
voidmain()
{
inti,j,n=0,x;/*n为标志变量*/
for(i=23;n==0;i+=2)/*控制试探的步长和过程*/
{
for(j=1,x=i;j<=4&&x>=11;j++)/*完成出售四次的操作*/
if((x+1)%(j+1)==0)/*若满足整除条件则进行实际的出售操作*/
x-=(x+1)/(j+1);
else{x=0;break;}/*否则停止计算过程*/
if(j==5&&x==11)/*若第四次余下11条则满足题意*/
{
printf("Thereare%dfishesatfirst.\\n",i);/*输出结果*/
n=1;/*控制退出试探过程*/
}
}
}
*运行结果
Thereare59fishesatfirst.
11.分数四则运算
对输入的两个分数进行+、-、*、/四则运算,输出分数结果。
算法分析如下:
对分数b/a与d/c,不管哪一种运算,其运算结果均为y/x形式。
对结果y/x进行化简,约去分子分母的公因数:
试用i(i=1,