C语言经典算法大全.pdf
《C语言经典算法大全.pdf》由会员分享,可在线阅读,更多相关《C语言经典算法大全.pdf(131页珍藏版)》请在淘文阁 - 分享文档赚钱的网站上搜索。
1、C C 语言经典算法大全语言经典算法大全老掉牙河内塔费式数列巴斯卡三角形三色棋老鼠走迷官(一)老鼠走迷官(二)骑士走棋盘八个皇后八枚银币生命游戏字串核对双色、三色河内塔背包问题(Knapsack Problem)数、运算蒙地卡罗法求 PIEratosthenes筛选求质数超长整数运算(大数运算)长 PI最大公因数、最小公倍数、因式分解完美数阿姆斯壮数最大访客数中序式转后序式(前序式)后序式的运算关于赌博洗扑克牌(乱数排列)Craps赌博游戏约瑟夫问题(Josephus Problem)集合问题排列组合格雷码(Gray Code)产生可能的集合1/131m元素集合的n个元素子集数字拆解排序得分排
2、行选择、插入、气泡排序Shell 排序法-改良的插入排序Shaker 排序法-改良的气泡排序Heap 排序法-改良的选择排序快速排序法(一)快速排序法(二)快速排序法(三)合并排序法基数排序法搜寻循序搜寻法(使用卫兵)二分搜寻法(搜寻原则的代表)插补搜寻法费氏搜寻法矩阵稀疏矩阵多维矩阵转一维矩阵上三角、下三角、对称矩阵奇数魔方阵4N 魔方阵2(2N+1)魔方阵2/1311.1.河内之塔河内之塔说明说明河内之塔(Towers of Hanoi)是法国人M.Claus(Lucas)于1883年从泰国带至法国的,河内为越战时北越的首都,即现在的胡志明市;1883年法国数学家 Edouard Luca
3、s曾提及这个故事,据说创世纪时Benares有一座波罗教塔,是由三支钻石棒(Pag)所支撑,开始时神在第一根棒上放置64个由上至下依由小至大排列的金盘(Disc),并命令僧侣将所有的金盘从第一根石棒移至第三根石棒,且搬运过程中遵守大盘子在小盘子之下的原则,若每日仅搬一个盘子,则当盘子全数搬运完毕之时,此塔将毁损,而也就是世界末日来临之时。解法解法如果柱子标为ABC,要由A搬至C,在只有一个盘子时,就将它直接搬至C,当有两个盘子,就将B当作辅助柱。如果盘数超过2个,将第三个以下的盘子遮起来,就很简单了,每次处理两个盘子,也就是:A-B、A-C、B-C这三个步骤,而被遮住的部份,其实就是进入程式的
4、递回处理。事实上,若有n个盘子,则移动完毕所需之次数为2n-1,所以当盘数为64时,则所需次数为:2-1=184467445为5.782e+16年,也就是约5000世纪,如果对这数字没什幺概念,就假设每秒钟搬一个盘子好了,也要约5850亿年左右。#include void hanoi(int n,char A,char B,char C)if(n=1)printf(Move sheet%d from%c to%cn,n,A,C);else hanoi(n-1,A,C,B);printf(Move sheet%d from%c to%cn,n,A,C);hanoi(n-1,B,A,C);int
5、main()int n;printf(请输入盘数:);scanf(%d,&n);hanoi(n,A,B,C);return 0;643/1312.Algorithm Gossip:2.Algorithm Gossip:费式数列费式数列说明说明Fibonacci为1200年代的欧洲数学家,在他的着作中曾经提到:若有一只免子每个月生一只小免子,一个月后小免子也开始生产。起初只有一只免子,一个月后就有两只免子,二个月后有三只免子,三个月后有五只免子(小免子投入生产).。如果不太理解这个例子的话,举个图就知道了,注意新生的小免子需一个月成长期才会投入生产,类似的道理也可以用于植物的生长,这就是Fibo
6、nacci数列,一般习惯称之为费氏数列,例如以下:1、1、2、3、5、8、13、21、34、55、89.解法解法依说明,我们可以将费氏数列定义为以下:fn=fn-1+fn-2fn=fn-1+fn-2if n 1if n 1fn=nfn=nif n=0,1if n=0,1#include#include#define N 20int main(void)int FibN=0;int i;Fib0=0;Fib1=1;for(i=2;i N;i+)Fibi=Fibi-1+Fibi-2;for(i=0;i N;i+)printf(%d,Fibi);printf(n);return 0;4/1313.3
7、.巴斯卡三角形巴斯卡三角形#include#define N 12long combi(int n,int r)int i;long p=1;for(i=1;i=r;i+)p=p*(n-i+1)/i;return p;void main()int n,r,t;for(n=0;n=N;n+)for(r=0;r=n;r+)int i;/*排版设定开始*/if(r=0)for(i=0;i=(N-n);i+)printf();else printf();/*排版设定结束*/printf(%3d,combi(n,r);printf(n);5/1314.Algorithm Gossip:4.Algorit
8、hm Gossip:三色棋三色棋说明说明三色旗的问题最早由E.W.Dijkstra所提出,他所使用的用语为Dutch Nation Flag(Dijkstra为荷兰人),而多数的作者则使用Three-Color Flag来称之。假设有一条绳子,上面有红、白、蓝三种颜色的旗子,起初绳子上的旗子颜色并没有顺序,您希望将之分类,并排列为蓝、白、红的顺序,要如何移动次数才会最少,注意您只能在绳子上进行这个动作,而且一次只能调换两个旗子。解法解法在一条绳子上移动,在程式中也就意味只能使用一个阵列,而不使用其它的阵列来作辅助,问题的解法很简单,您可以自己想像一下在移动旗子,从绳子开头进行,遇到蓝色往前移,
9、遇到白色留在中间,遇到红色往后移,如下所示:只是要让移动次数最少的话,就要有些技巧:如果图中W所在的位置为白色,则W+1,表示未处理的部份移至至白色群组。如果W部份为蓝色,则B与W的元素对调,而B与W必须各+1,表示两个群组都多了一个元素。如果W所在的位置是红色,则将W与R交换,但R要减1,表示未处理的部份减1。注意B、W、R并不是三色旗的个数,它们只是一个移动的指标;什幺时候移动结束呢?一开始时未处理的R指标会是等于旗子的总数,当R的索引数减至少于W的索引数时,表示接下来的旗子就都是红色了,此时就可以结束移动,如下所示:#include#include#include#define BLUE
10、 b#define WHITE w#define RED r#define SWAP(x,y)char temp;temp=colorx;colorx=colory;colory=temp;int main()char color=r,w,b,w,w,6/131b,r,b,w,r,0;int wFlag=0;int bFlag=0;int rFlag=strlen(color)-1;int i;for(i=0;i strlen(color);i+)printf(%c,colori);printf(n);while(wFlag=rFlag)if(colorwFlag=WHITE)wFlag+;e
11、lse if(colorwFlag=BLUE)S,wFlag);bFlag+;wFlag+;else while(wFlag rFlag&colorrFlag=RED)rFlag-;S,wFlag);rFlag-;for(i=0;i strlen(color);i+)printf(%c,colori);printf(n);return 0;7/1315.Algorithm Gossip:5.Algorithm Gossip:老鼠走迷官(一)老鼠走迷官(一)说明说明老鼠走迷宫是递回求解的基本题型,我们在二维阵列中使用2表示迷宫墙壁,使用1来表示老鼠的行走路径,试以程式求出由入口至出口的路径。解法
12、解法老鼠的走法有上、左、下、右四个方向,在每前进一格之后就选一个方向前进,无法前进时退回选择下一个可前进方向,如此在阵列中依序测试四个方向,直到走到出口为止,这是递回的基本题,请直接看程式应就可以理解。#include#include int visit(int,int);int maze77=2,2,2,2,2,2,2,2,0,0,0,0,0,2,2,0,2,0,2,0,2,2,0,0,2,0,2,2,2,2,0,2,0,2,2,2,0,0,0,0,0,2,2,2,2,2,2,2,2;int startI=1,startJ=1;/入口int endI=5,endJ=5;/出口int succ
13、ess=0;int main(void)int i,j;printf(显示迷宫:n);for(i=0;i 7;i+)for(j=0;j 7;j+)if(mazeij=2)printf();elseprintf();printf(n);if(visit(startI,startJ)=0)printf(n没有找到出口!n);8/131else printf(n显示路径:n);for(i=0;i 7;i+)for(j=0;j 7;j+)if(mazeij=2)printf();else if(mazeij=1)printf();elseprintf();printf(n);return 0;int
14、visit(int i,int j)mazeij=1;if(i=endI&j=endJ)success=1;if(success!=1&mazeij+1=0)visit(i,j+1);if(success!=1&mazei+1j=0)visit(i+1,j);if(success!=1&mazeij-1=0)visit(i,j-1);if(success!=1&mazei-1j=0)visit(i-1,j);if(success!=1)mazeij=0;return success;9/1316.Algorithm Gossip:6.Algorithm Gossip:老鼠走迷官(二)老鼠走迷官
15、(二)说明说明由于迷宫的设计,老鼠走迷宫的入口至出口路径可能不只一条,如何求出所有的路径呢?解法解法求所有路径看起来复杂但其实更简单,只要在老鼠走至出口时显示经过的路径,然后退回上一格重新选择下一个位置继续递回就可以了,比求出单一路径还简单,我们的程式只要作一点修改就可以了。#include#include void visit(int,int);int maze99=2,2,2,2,2,2,2,2,2,2,0,0,0,0,0,0,0,2,2,0,2,2,0,2,2,0,2,2,0,2,0,0,2,0,0,2,2,0,2,0,2,0,2,0,2,2,0,0,0,0,0,2,0,2,2,2,0,
16、2,2,0,2,2,2,2,0,0,0,0,0,0,0,2,2,2,2,2,2,2,2,2,2;int startI=1,startJ=1;/入口int endI=7,endJ=7;/出口int main(void)int i,j;printf(显示迷宫:n);for(i=0;i 7;i+)for(j=0;j 7;j+)if(mazeij=2)printf();elseprintf();printf(n);visit(startI,startJ);10/131return 0;void visit(int i,int j)int m,n;mazeij=1;if(i=endI&j=endJ)pr
17、intf(n显示路径:n);for(m=0;m 9;m+)for(n=0;n 9;n+)if(mazemn=2)printf();else if(mazemn=1)printf();elseprintf();printf(n);if(mazeij+1=0)visit(i,j+1);if(mazei+1j=0)visit(i+1,j);if(mazeij-1=0)visit(i,j-1);if(mazei-1j=0)visit(i-1,j);mazeij=0;11/1317.Algorithm Gossip:7.Algorithm Gossip:骑士走棋盘骑士走棋盘说明说明骑士旅游(Knight
18、 tour)在十八世纪初倍受数学家与拼图迷的注意,它什么时候被提出已不可考,骑士的走法为西洋棋的走法,骑士可以由任一个位置出发,它要如何走完 所有的位置?解法解法骑士的走法,基本上可以使用递回来解决,但是纯綷的递回在维度大时相当没有效率,一个聪明的解法由J.C.Warnsdorff在1823年提出,简单的说,先将最难的位置走完,接下来的路就宽广了,骑士所要走的下一步,为下一步再选择时,所能走的步数最少的一步。,使用这个方法,在不使用递回的情况下,可以有较高的机率找出走法(找不到走法的机会也是有的)。#include int board88=0;int main(void)int startx,
19、starty;int i,j;printf(输入起始点:);scanf(%d%d,&startx,&starty);if(travel(startx,starty)printf(游历完成!n);else printf(游历失败!n);for(i=0;i 8;i+)for(j=0;j 8;j+)printf(%2d,boardij);putchar(n);return 0;int travel(int x,int y)/对应骑士可走的八个方向int ktmove18=-2,-1,1,2,2,1,-1,-2;int ktmove28=1,2,2,1,-1,-2,-2,-1;12/131/测试下一步
20、的出路int nexti8=0;int nextj8=0;/记录出路的个数int exists8=0;int i,j,k,m,l;int tmpi,tmpj;int count,min,tmp;i=x;j=y;boardij=1;for(m=2;m=64;m+)for(l=0;l 8;l+)existsl=0;l=0;/试探八个方向for(k=0;k 8;k+)tmpi=i+ktmove1k;tmpj=j+ktmove2k;/如果是边界了,不可走if(tmpi 0|tmpj 7|tmpj 7)continue;/如果这个方向可走,记录下来if(boardtmpitmpj=0)nextil=tm
21、pi;nextjl=tmpj;/可走的方向加一个l+;count=l;/如果可走的方向为0个,返回if(count=0)13/131return 0;else if(count=1)/只有一个可走的方向/所以直接是最少出路的方向min=0;else/找出下一个位置的出路数for(l=0;l count;l+)for(k=0;k 8;k+)tmpi=nextil+ktmove1k;tmpj=nextjl+ktmove2k;if(tmpi 0|tmpj 7|tmpj 7)continue;if(boardtmpitmpj=0)existsl+;tmp=exists0;min=0;/从可走的方向中寻
22、找最少出路的方向for(l=1;l count;l+)if(existsl tmp)tmp=existsl;min=l;/走最少出路的方向i=nextimin;j=nextjmin;boardij=m;return 1;14/1318.Algorithm Gossip:8.Algorithm Gossip:八皇后八皇后说明说明西洋棋中的皇后可以直线前进,吃掉遇到的所有棋子,如果棋盘上有八个皇后,则这八个皇后如何相安无事的放置在棋盘上,1970年与1971年,E.W.Dijkstra与N.Wirth曾经用这个问题来讲解程式设计之技巧。解法解法关于棋盘的问题,都可以用递回求解,然而如何减少递回的次
23、数?在八个皇后的问题中,不必要所有的格子都检查过,例如若某列检查过,该该列的其它格子就不用再检查了,这个方法称为分支修剪。#include#include#define N 8int columnN+1;/同栏是否有皇后,1表示有int rup2*N+1;/右上至左下是否有皇后int lup2*N+1;/左上至右下是否有皇后int queenN+1=0;int num;/解答编号void backtrack(int);/递回求解int main(void)int i;num=0;for(i=1;i=N;i+)columni=1;for(i=1;i=2*N;i+)rupi=lupi=1;back
24、track(1);return 0;void showAnswer()int x,y;15/131printf(n解答%dn,+num);for(y=1;y=N;y+)for(x=1;x N)showAnswer();else for(j=1;j=N;j+)if(columnj=1&rupi+j=1&lupi-j+N=1)queeni=j;/设定为占用columnj=rupi+j=lupi-j+N=0;backtrack(i+1);columnj=rupi+j=lupi-j+N=1;16/1319.Algorithm Gossip:9.Algorithm Gossip:八枚银币八枚银币说明说明
25、现有八枚银币a b c d e f g h,已知其中一枚是假币,其重量不同于真币,但不知是较轻或较重,如何使用天平以最少的比较次数,决定出哪枚是假币,并得知假币比真币较轻或较重。解法解法单就求假币的问题是不难,但问题限制使用最少的比较次数,所以我们不能以单纯的回圈比较来求解,我们可以使用决策树(decision tree),使用分析与树状图来协助求解。一个简单的状况是这样的,我们比较a+b+c与d+e+f,如果相等,则假币必是 g或h,我们先比较g或h哪个较重,如果g较重,再与a比较(a是真币),如果g等于a,则g为真币,则h为假币,由于h比g轻而 g是真币,则h假币的重量比真币轻。#incl
- 配套讲稿:
如PPT文件的首页显示word图标,表示该PPT已包含配套word讲稿。双击word图标可打开word文档。
- 特殊限制:
部分文档作品中含有的国旗、国徽等图片,仅作为作品整体效果示例展示,禁止商用。设计者仅对作品中独创性部分享有著作权。
- 关 键 词:
- 语言 经典 算法 大全
限制150内