第6章 程序设计的问题求解基础_第1页
第6章 程序设计的问题求解基础_第2页
第6章 程序设计的问题求解基础_第3页
第6章 程序设计的问题求解基础_第4页
第6章 程序设计的问题求解基础_第5页
已阅读5页,还剩44页未读 继续免费阅读

下载本文档

版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领

文档简介

第6章程序设计的问题求解基础哈尔滨工业大学6.1问题求解策略算法设计的基本思路就是寻找解决问题的规律。指在数学思想支持下的解题思路、方式和方法。常用的问题求解策略枚举、递推、迭代、分治、递归6.2枚举——从找回你的密码谈起穷举法(Exhaustion),也称枚举法(Enumeration)列举所有可能,逐一试探基本思想根据问题的部分已知条件预估解的范围在此范围内对所有可能的情况进行逐一验证若某个情况符合题目的全部条件,则该情况为本问题的一个解若全部情况的验证结果均不符合题目的全部条件,则说明该题无解直到找到满足已知条件的解为止6.2枚举——从找回你的密码谈起确定穷举对象和穷举范围确定判定条件

符合答案的条件分支结构实现

影响算法的时间复杂度循环结构实现for(穷举范围){……

if(符合答案的条件)……}枚举法求解问题的两个基本要素【例6.1】百鸡问题是我国古代数学名著《张丘建算经》中的一个著名数学问题,它给出了由三个未知量的两个方程组成的不定方程组的解。《张丘建算经》为北魏张丘建所著,所载问题大部分为当时社会生活中的实际问题,如有关测量、纺织、交换、纳税、冶炼、土木工程等,涉及面广泛,其问题和解法均超出《九章算术》。百鸡问题具体是这样的:鸡翁一,值钱五;鸡母一,值钱三;鸡雏三,值钱一。百钱买百鸡,问鸡翁、母、雏各几何?其意为:公鸡每只5元,母鸡每只3元,小鸡3只1元。请用穷举法编程计算,若用100元买100只鸡,则公鸡、母鸡和小鸡各能买多少只。6.2枚举——从找回你的密码谈起6.2枚举——从找回你的密码谈起#include<stdio.h>intmain(void){intx,y,z;for(x=0;x<=100;x++){for(y=0;y<=100;y++){for(z=0;z<=100;z++){if(x+y+z==100&&5*x+3*y+z/3.0==100)printf("x=%d,y=%d,z=%d\n",x,y,z);}}}return0;}6.2枚举——从找回你的密码谈起#include<stdio.h>intmain(void){intx,y,z;for(x=0;x<=20;x++){for(y=0;y<=33;y++){z=100-–x-y;if(5*x+3*y+z/3.0==100)printf("x=%d,y=%d,z=%d\n",x,y,z);}}return0;}【例6.3】韩信点兵问题。西汉开国功臣、军事家韩信有一队兵,他想知道有士兵多少人,便让士兵排队报数。按从1至5报数,最末一个士兵报的数为1;按从1至6报数,最末一个士兵报的数为5;按从1至7报数,最末一个士兵报的数为4;最后再按从1至11报数,最末一个士兵报的数为10。请问韩信至少有多少兵。确定问题的输入和输出输入:无;输出:士兵至少x人确定穷举对象:士兵数x确定搜索范围:x从1变化……确定判定条件:x被5、6、7、11整除余数为1、5、4、10x%5==1&&x%6==5&&x%7==4&&x%11==106.2枚举——从找回你的密码谈起6.2枚举——从找回你的密码谈起#include<stdio.h>intmain(void){intx;for(x=1;x<3000;x++){if(x%5==1&&x%6==5&&x%7==4&&x%11==10){printf("x=%d\n",x);}}return0;}#include<stdio.h>intmain(void){intx;for(x=1;;x++){if(x%5==1&&x%6==5&&x%7==4&&x%11==10){printf("x=%d\n",x);break;}}return0;}6.2枚举——从找回你的密码谈起#include<stdio.h>#include<stdlib.h>intmain(void){intx;for(x=1;;x++){if(x%5==1&&x%6==5&&x%7==4&&x%11==10){printf("x=%d\n",x);exit(0);}}return0;}#include<stdio.h>intmain(void){intx;intfind=0;//置找到标志变量为假

for(x=1;!find;x++)//find为假时继续循环

{

if(x%5==1&&x%6==5&&x%7==4&&x%11==10){printf("x=%d\n",x);find=1;//置找到标志变量为真

}}

return0;}6.2枚举——从找回你的密码谈起#include<stdio.h>intmain(void){ intx=0; intfind=0; do{ x++; find=(x%5==1&&x%6==5&&x%7==4&&x%11==10); }while(!find); printf("x=%d\n",x);return0;}#include<stdio.h>intmain(void){ intx=0;//因do-while循环中先对x加1,故这里x初始化为0

do{ x++; }while(!(x%5==1&&x%6==5&&x%7==4&&x%11==10)); printf("x=%d\n",x);return0;}【例6.4】还原算术表达式。请编写一个程序求解以下算式中各字母所代表的数字的值,已知不同的字母代表不同的数字。先从键盘输入小于1000的n值,如果n不小于1000,则重新输入n值,然后输出第一个满足条件的解。6.2枚举——从找回你的密码谈起#include<stdio.h>intmain(void){intx,y,z,n,find=0;do{printf("Inputn(n<1000):");scanf("%d",&n);}while(n>=1000);//若输入的n不在合法范围内,则重新输入

for(x=1;x<=9;++x){for(y=1;y<=9;++y){for(z=0;z<=9;++z){if(100*x+10*y+z+z+10*z+100*y==n&&x!=y&&y!=z&&x!=z){printf("X=%d,Y=%d,Z=%d\n",x,y,z);find=1;}}}}if(!find)printf("Notfound!");return0;}【例6.4】还原算术表达式。请编写一个程序求解以下算式中各字母所代表的数字的值,已知不同的字母代表不同的数字。先从键盘输入小于1000的n值,如果n不小于1000,则重新输入n值,然后输出第一个满足条件的解。6.2枚举——从找回你的密码谈起#include<stdio.h>intmain(void){

……

for(x=1;x<=9&&!find;++x){for(y=1;y<=9&&!find;++y){if(x==y)continue;//若y与x相等,就直接换下一个y再试

for(z=0;z<=9&&!find;++z){if(y==z||x==z)continue;//若z与x或y相等,就换下一个zif(100*x+10*y+z+z+10*z+100*y==n){printf("X=%d,Y=%d,Z=%d\n",x,y,z);find=1;}count++;}}}printf("Notfound!");return0;}【例6.4】还原算术表达式。请编写一个程序求解以下算式中各字母所代表的数字的值,已知不同的字母代表不同的数字。先从键盘输入小于1000的n值,如果n不小于1000,则重新输入n值,然后输出第一个满足条件的解。6.2枚举——从找回你的密码谈起#include<stdio.h>intmain(void){

……

for(x=1;x<=9;++x){for(y=1;y<=9;++y){if(x==y)continue;for(z=0;z<=9;++z){if(y==z||x==z)continue;count++;if(100*x+10*y+z+z+10*z+100*y==n){printf("X=%d,Y=%d,Z=%d\n",x,y,z);find=1;gotoEND;}}}}if(!find)printf("Notfound!");END:printf("count=%d\n",count);return0;}【例6.5】请编写一个程序,计算并输出1~n之间的所有素数之和。6.2枚举——从找回你的密码谈起intIsPrime(intm){inti;intsquareRoot=(int)sqrt(m);if(m<=1){return0;//负数、0和1都不是素数

}

for(i=2;i<=squareRoot;++i){if(m%i==0){gotoEND;}}END:returni<=squareRoot?0:1;}intIsPrime(intm){inti;intsquareRoot=(int)sqrt(m);if(m<=1){return0;//负数、0和1都不是素数

}

for(i=2;i<=squareRoot;++i){if(m%i==0){break;}}returni<=squareRoot?0:1;}【例6.5】请编写一个程序,计算并输出1~n之间的所有素数之和。6.2枚举——从找回你的密码谈起intIsPrime(intm){intflag=1;intsquareRoot=(int)sqrt(m);if(m<=1){flag=0;//负数、0和1都不是素数

}

for(inti=2;i<=squareRoot&&flag;++i){if(m%i==0){flag=0;//若能被整除,则不是素数

}}

returnflag;}#include<stdbool.h>boolIsPrime(intm){boolflag=true;intsquareRoot=(int)sqrt(m);if(m<=1){flag=false;//负数、0和1都不是素数

}

for(inti=2;i<=squareRoot&&flag;++i){if(m%i==0){flag=false;//若能被整除,则不是素数

}}

returnflag;}intmain(void){intn;scanf("%d",&n);printf("sum=%d\n",SumofPrime(n));return0;}【例6.6】德国人哥德巴赫在1742年提出了自己无法证明的两个猜想:(1)每个大于2的偶数都是两个素数之和;(2)每个大于5的奇数都是三个素数之和。1973年,我国数学家陈景润攻克了数学界200多年悬而未决的世界级数学难题即“哥德巴赫猜想”中的“1+2”,成为哥德巴赫猜想研究史上的里程碑。他的论文发表后立即在国际数学界引起了轰动,被公认为是对哥德巴赫猜想研究的重大贡献。他的成果被国际数学界称为“陈氏定理”,写进美、英、法、苏、日等六国的许多数论书中。2009年,陈景润被评为100位新中国成立以来感动中国人物之一。现在,请你编写一个程序来验证:任何一个大于或等于6但不超过2000000000的足够大的偶数n总能表示为两个素数之和。例如,8=3+5,12=5+7等等。如果n符合“哥德巴赫猜想”,则输出将n分解为两个素数之和的等式,否则输出“n不符合哥德巴赫猜想!”的提示信息。他曾经说过:时间是个常数,花掉一天等于浪费24小时;学习要有三心:信心、决心、恒心;攀登科学高峰,就像登山运动员攀登珠穆朗玛峰一样,要克服无数艰难险阻,懦夫和懒汉是不可能享受到胜利的喜悦和幸福的。6.2枚举——从找回你的密码谈起【例6.5】请编写一个程序,计算并输出1~n之间的所有素数之和。6.2枚举——从找回你的密码谈起#include<stdio.h>#include<math.h>#include<stdbool.h>boolIsPrime(longm);boolGoldbach(longn);intmain(void){longn;intret;do{printf("Inputn:");ret=scanf("%ld",&n);if(ret!=1)while(getchar()!='\n');}while(ret!=1||n%2!=0||n<6||n>2000000000);if(!Goldbach(n)){printf("%ld不符合哥德巴赫猜想\n",n);}return0;}【例6.5】请编写一个程序,计算并输出1~n之间的所有素数之和。6.2枚举——从找回你的密码谈起boolGoldbach(longn){boolfind=false;for(longa=3;a<=n/2&&!find;a+=2){if(IsPrime(a)&&IsPrime(n-a)){printf("%ld=%ld+%ld",n,a,n-a);find=true;}}returnfind;}boolIsPrime(longm){boolflag=true;intsquareRoot=(int)sqrt(m);if(m<=1)flag=false;//负数、0和1都不是素数

for(inti=2;i<=squareRoot&&flag;++i){if(m%i==0)flag=false;//若能被整除,则不是素数

}

returnflag;}6.4递推——荷花定律和大自然中的秘密在一个荷花池里,第一天荷花开放的很少,第二天开放的数量是第一天的两倍,之后的每一天,荷花都会以前一天两倍的数量开放。如果到第30天,荷花就开满了整个池塘,那么请问荷花是在第几天开了半个池塘呢?向日葵的花盘中有2组对数螺线,一组顺时针方向盘绕,另一组则逆时针方向盘绕,并且彼此相嵌。虽然不同的向日葵品种中,这些顺逆螺旋的数目并不固定,但往往不会超出34和55、55和89、89和144这三组数字。除了向日葵,松果也符合这一奇妙的自然规律植物的花瓣、萼片、果实的数目以及其他方面的特征,都非常吻合一个奇特的数列,即Fibonacci数列:1,1,2,3,5,8,13,21,34,55,89......。这个数列从第三项开始,每一项都是前两项之和6.4递推——荷花定律和大自然中的秘密正向顺推,就是从已知条件出发,向着所求问题前进,最后与所求问题联系起来例如,已知一个汉堡包12元,一个蛋挞比一个汉堡包贵5元,问买两种各一个需多少元?首先根据已知条件可以推出一个蛋挞的价格是12+5=17元,然后可以推出买两种各一个需17+12=29。反向逆推,即从所求问题出发,向着已知条件靠拢,最后与已知条件联系起来。例如,小明来到一家早餐店,拿出一半钱吃早餐,又花了3.5元钱买饮料,还剩1元钱,问他原来带了多少钱?这个问题适合使用反向逆推的方法,即从问题的结果出发,一步一步还原出答案。因为我们知道小明现在有1元钱,他做的最后一件事是花3.5元买饮料,所以我们可以把3.5元和他的一元加起来,逆推出他买饮料之前有4.5元,根据已知条件,他花一半钱吃早餐还剩4.5元,那么他吃早餐一定花了4.5元,那么原来他就应该有9元。可递推求解的问题一般具有以下两个特点:(1)问题可以划分成多个状态;(2)除初始状态外,其它各个状态都可以用固定的递推关系式来表示。6.4.1正向顺推实例11

2

3

5

8

......f1

f2f3

f1

f2f3

f1

f2f3

f1

f2f3

f3=f1+f2;f1=f2;f2=f3;11

2

3

5

8

......f1

f2

f1

f2

f1

f2

f1

f1=f1+f2;f2=f2+f1;【例6.7】计算Fibonacci数列的前n项。6.4.1正向顺推实例【例6.7】计算Fibonacci数列的前n项。#include<stdio.h>longFib(intn);intmain(void){inti;longn;printf("Inputn:");scanf("%ld",&n);for(i=1;i<=n;i++){printf("%4ld",Fib(i));}return0;}//函数功能:正向顺推法计算并返回Fibonacci数列的第n项longFib(intn){inti;longf1=1,f2=1,f3=2;if(n==1){return1;}elseif(n==2){return1;}else{for(i=3;i<=n;i++)//每递推一次计算一项

{

f3=f1+f2;f1=f2;f2=f3;}returnf3;}}6.4.1正向顺推实例【例6.7】计算Fibonacci数列的前n项。#include<stdio.h>longFib(intn);intmain(void){inti;longn;printf("Inputn:");scanf("%ld",&n);for(i=1;i<=n;i++){printf("%4ld",Fib(i));}return0;}//函数功能:正向顺推法计算并返回Fibonacci数列的第n项longFib(intn){inti;longf1=1,f2=1;if(n==1){return1;}elseif(n==2){return1;}else{for(i=1;i<(n+1)/2;i++)

{

f1=f1+f2;f2=f2+f1;}returnn%2!=0?f1:f2;}}6.4.2反向逆推实例【例6.8】猴子吃桃问题猴子第一天摘下若干个桃子,吃了一半,还不过瘾,又多吃了一个。第二天早上又将剩下的桃子吃掉一半,并且又多吃了一个。以后每天早上都吃掉前一天剩下的一半零一个。到第10天早上再想吃时,发现只剩下一个桃子。问第一天共摘了多少桃子。每天剩下的桃子数比前一天的一半少一个每天剩下的桃子数加1之后,刚好是前一天的一半天数:

10

9

8

7

6

5

4

3

2

1

4

10

22

46

94

190

382

766

1534

6.4.2反向逆推实例#include<stdio.h>intMonkeyEatPeach(intdayn);intmain(void){intdays,total;printf("Inputdays:");scanf("%d",&days);total=MonkeyEatPeach(days);printf("x=%d\n",total);return0;}intMonkeyEatPeach(intday){intx=1;do{x=(x+1)*2;day--;}while(day>1);returnx;}intMonkeyEatPeach(intday){intx=1;while(day>1){x=(x+1)*2;day--;}returnx;}6.5递归——千里之行始于足下的启示6.5.1递归的内涵与数学基础一种问题求解的策略假设你已经走了999步,那么你再走一步即可迈出第1000步假设你已经走了998步,那么你再走一步即可迈出第999步……直到你迈出第1步,递归结束walkwalk1“从前有座山,山里有座庙,庙里有个老和尚给小和尚讲故事,讲什么呢?”“从前有座山,山里有座庙,庙里有个老和尚给小和尚讲故事,讲什么呢?”“从前有座山,山里有座庙,庙里有个老和尚给小和尚讲故事,讲什么呢?”………….6.5.1递归的内涵与数学基础如果一个对象部分地由它自己组成或按它自己定义,则称它是递归的(Recursive)递归(Recursion)函数/过程/子程序在运行过程中直接或间接调用自身而产生的重入现象6.5.1递归的内涵与数学基础递归算法必须包含如下两个部分:由其自身定义的与原始问题类似的更小规模的子问题,它使递归过程持续进行,称为一般条件(Generalcase)所描述问题的最简单的情况,它是一个能控制递归过程结束的条件,称为基本条件(Basecase)计算n!问题的一般条件可以用递归公式表示为:n!=n×(n-1)!(当n>1时)计算n!问题的基本条件可以表示为:n!=1(当n=1时)6.4.2递归函数的基本要素if(递归终止条件)//基本条件,也称边界条件,代表递归的出口return递归公式的初值;else//一般条件,表示递推关系return递归函数调用返回的结果值;

【例6.9】分别用迭代和递归两种方法计算正整数n的阶乘即n!。326.4.2递归函数的基本要素#include<stdio.h>unsignedintFact(unsignedintn);intmain(void){unsignedintn;printf("Inputn:");scanf("%u",&n);printf("%u!=%u\n",n,Fact(n));return0;}//函数功能:用迭代法计算n的阶乘并返回unsignedintFact(unsignedintn){unsignedinti,p=1; for(i=2;i<=n;i++) { p=p*i; }returnp;}#include<stdio.h>unsignedintFact(unsignedintn);intmain(void){unsignedintn;printf("Inputn:");scanf("%u",&n);printf("%u!=%u\n",n,Fact(n));return0;}//函数功能:用递归法计算n的阶乘并返回unsignedintFact(unsignedintn){if(n==1||n==0)//基本条件

{

return1;}else//一般条件

{

returnn*Fact(n-1);//递归调用

}}#include<stdio.h>intMonkeyEatPeach(intdays);intmain(){ intx,days;printf("Inputdays:"); scanf("%d",&days); x=MonkeyEatPeach(days); printf("x=%d\n",x); return0;}intMonkeyEatPeach(intdays)//由第days天推出第days-1天{if(days==1)//递归结束条件

return1;//递归初值else return2*(MonkeyEatPeach(days-1)+1);//递推式}6.4.2递归函数的基本要素【例6.10】采用递归法编写求解例6.8的猴子吃桃问题的程序代码6.5.3递归的执行过程递归算法的执行过程一般可分为两个阶段:递推阶段(也称递归前进阶段)把一个较复杂的问题逐步分解为与原始问题类似的更小规模的子问题,逐步简化直至最终转化为一个最简单的问题,可以直接求解并终止递归回归阶段(也称递归返回阶段)当获得最简单情况的解后,逐级返回,依次得到稍复杂问题的解。6.5.3递归的执行过程栈操作“后进先出”的特点使其能够精确地满足函数调用和返回的顺序,因此特别适合于保存与函数调用相关的信息。保存这些信息的栈空间,称为活跃记录或者栈帧。栈帧中主要存储输入参数、计算表达式时用到的临时变量的值、函数调用时保存的状态信息(如返回地址)等6.5.4递归与迭代的关系递归的本质不断地把规模较大的问题分解成与原来问题相同但规模较小的同类子问题,直到这个小的问题能够直接使用简单的方法解决为止子问题与原问题的区别只是问题的规模不同而已递归和迭代的关系迭代显式地使用重复结构,而递归使用选择结构,通过重复函数调用实现重复结构迭代和递归都涉及终止测试,迭代在循环测试条件为假时终止循环,递归则在遇到基本条件时终止递归。迭代不断修改循环计数变量,直到它使循环条件为假时为止。递归则不断产生最初问题的简化版本,直到简化为递归的基本条件。6.5.4递归与迭代的关系【例6.11】最大公约数问题。两个正整数的最大公约数(GreatestCommonDivisor,GCD)是能够整除这两个整数的最大整数。从键盘任意输入两个正整数a和b,请分别采用枚举法、欧几里德算法(辗转相除法)以及更相减损术三种不同的方法编程计算并输出两个正整数a和b的最大公约数。(1)穷举法

intGcd(inta,intb){ inti,t; if(a<=0||b<=0) return-1; t=a<b?a:b;

for(i=t;i>0;i--) { if(a%i==0&&b%i==0) returni; } return1;}

6.5.4递归与迭代的关系(2)欧几里德算法(辗转相除法)

对正整数a和b,连续进行求余运算

r=a%b直到余数r为0为止

此时非0的除数,即为所求

若r≠0,则b作为新的a,r作为新的b即Gcd(a,b)=Gcd(b,r)重复a%b运算,直到r=0时为止。

intGcd(inta,intb){ intr; if(a<=0||b<=0) { return-1; }

do{ r=a%b; a=b; b=r; }while(r!=0);

returna;}

6.5.4递归与迭代的关系辗转相除法的递归实现

intGcd(inta,intb){ if(a<=0||b<=0) { return-1; }intr=a%b;

if(r==0)returnb;elsereturnGcd(b,r);}

6.5.4递归与迭代的关系(3)更相减损术(《九章算术》)——递归方法性质1如果a>b,则Gcd(a,b)=Gcd(a-b,b)性质2如果b>a,则Gcd(a,b)=Gcd(a,b-a)性质3如果a=b,则Gcd(a,b)=a=b

intGcd(inta,intb){if(a<=0||b<=0)return-1;

if(a==b)returna;elseif(a>b)returnGcd(a-b,b);elsereturnGcd(a,b-a);}《九章算术》是我国古代的数学专著,值得我们后人为之骄傲和自豪的是,作为一部世界数学名著,《九章算术》早在隋唐时期即已传入朝鲜、日本。它已被译成日、俄、德、法等多种文字版本。《九章算术》是算经十书中最重要的一种,该书系统地总结了战国、秦、汉时期的数学成就,是当时世界上最简练有效的应用数学,它的出现标志中国古代数学形成了完整的体系。更相减损术就是《九章算术》中记载的一种求最大公约数的方法,其主要思想是

6.5.4递归与迭代的关系更相减损术——迭代方法

intGcd(inta,intb){ if(a<=0||b<=0) { return-1; }

while(a!=b)

{ if(a>b) { a=a-b; } elseif(b>a) {

b=b-a; } }

returna;}6.5.4递归与迭代的关系【例6.12】利用如下递归公式,用递归法编程计算并输出Fibonacci数列的前n项。longFib(intn){if(n==1)//基本条件

{return1;}elseif(

温馨提示

  • 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
  • 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
  • 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
  • 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
  • 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
  • 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
  • 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。

评论

0/150

提交评论