循环赛日程表
八个队循环赛轮次排
19日下午
13:00-14:00
第二场地
劳申学
张启波
联队-三区
14:00-15:00
第二场地
劳申学
汉春雷
社区-六区
第三场地
杜继文
张启波
五区-四区
15:00-16:00
第二场地
姜凤鸣
汉春雷
二区-三区
16:00-17:00
第二场地
劳申学
汉春雷
七区-四区
17:00-18:00
第二场地
张启波
劳申学
(女队)六区-医院
第三场地
姜凤鸣
杜继文
联队-六区
20日上午
7:30-8:30
第二场地
张启波
汉春雷
二区-四区
第三场地
杜继文
姜凤鸣
五区-社区
8:30-9:30
第二场地
劳申学
张启波
三区-六区
9:30-10:30
第二场地
姜凤鸣
汉春雷
七区-社区
10:30-11:30
第二场地
姜凤鸣
劳申学
(女队)六区-学校
第三场地
鸭绿河农场第十届职工篮球赛
B组单循环赛日程表
参赛队
日期
时间
场地
裁判员
主裁
副裁
二区-联队
19日上午
8:30-9:30
第二场地
张启波
姜凤鸣
社区-三区
第三场地
劳申学
杜继文
五区-七区
9:30-10:30
第二场地
姜凤鸣
汉春雷
六区-四区
10:30-11:30
第二场地
张启波
网球循环赛日程表
一、问题表述:设有n个运动员要进行网球循环赛。
设计一个满足以下要求的比赛日程表,(1) 每个选手必须与其他n-1个选手各赛一次;(2) 每个选手一天只能赛一次;(3) 当n是偶数时,循环赛进行n-1天,当n是奇数时,循环赛进行n天二、分析问题题目是要n名运动员进行循环比赛。
当n为偶数时,正好每天都可以两两一组,与其余的n-1个选手比赛,只需n-1天;而当n为奇数,每天将有一个选手轮空,比赛将持续n天。
可以采用的算法如下:1.算法一:使用分治法当n为偶数时,可以讲问题分为两个部分n/2; 然后继续划分,知道最后剩余两名选手单独比赛。
当n为奇数时,增设一个虚拟选手,运动员为n+1个,将问题转化为是偶数的情形。
当选手与虚拟选手比赛时,表示轮空,因此只需要关注n为偶数的情形。
a)当n/2为偶数时,与n = 2^k情形类此。
b)当n/2为奇数时,增设一个虚拟的选手,递归返回的将有轮空的选手,可以讲在前面n/2轮比赛的选手与后面n/2轮空的选手进行比赛。
2.算法二:利用边是奇数的正多边形。
特点:以多边形中的任意一个顶点画对称轴,其余偶数对顶点相互对称。
N名选手编号为1~n,将其画成一个正多边形。
a)所以当n为奇数时,第一天1号休息,其余以一号为对称轴,两两对称打比赛,第二天开始一次轮流休息,其余一休息的那个人编号为对称轴,两两比赛。
这样比赛可进行n天。
如图:12345678012345678对称轴此时n=9,为奇数,从0开始每天有一个人轮空对称轴b) 当n 为偶数时,取出编号最大的,其他的组成一个正多边形,n 号一次顺序与1,2,。
n -1号选手比赛,其他与a )相同。
如图所示:(图中是从0开始编号)123456789 9N=2k 时9三、 理论分析算法及实现1. 算法一:使用分治法a) 算法的思路:按分治策略,可以将所有的选手对分为两组(如果n 是偶数,则直接分为n/2每组,如果n 是奇数,则取(n+1)/2每组),n 个选手的比赛日程表就可以通过为(n/2或(n+1)/2)个选手设计的比赛日程表来决定。
用C++编写循环赛日程表
循环赛日程表问题描述:设有n位选手参加网球循环赛,n=2^k,循环赛共进行n-1天,每位选手要与其他n-1位选手比赛一场,且每位选手每天比赛一场,不能轮空,按一下要求为比赛安排日程,(1)每位选手必须与其他n-1格选手格赛一场;(2)每个选手每天只能赛一场;(3)循环赛一共进行n-1天;#include<iostream.h>int a[50][50];void table (int x,int k)//此函数为从x号球员起的共2的k次方名球员的安排日程表{int i,j,y=1;if(k==1)//只有两名球员{a[x][0]=x;a[x][1]=x+1;a[x+1][0]=x+1;a[x+1][1]=x;}else{for(i=1;i<=k-1;i++){y=y*2;}table(x,k-1);table(x+y,k-1);for(i=x;i<x+y;i++){for(j=y;j<2*y;j++)a[i][j]=a[i+y][j-y];}for(i=x+y;i<x+2*y;i++){for(j=y;j<2*y;j++)a[i][j]=a[i-y][j-y];}}}void main(){int i,j,k;int n=1;cout<<"请输入k值"<<endl;cin>>k;for(i=1;i<=k;i++){n=n*2;}cout<<"参赛人数"<<" "<<n<<endl; table(1,k);cout<<"*****循环赛日程表****"<<endl;cout<<endl;cout<<"日期:";for( i=1;i<n;i++)cout<<" "<<i;cout<<endl;for(i=1;i<n;i++){cout<<endl;for(j=1;j<n;j++)cout<<" "<<a[i][j]<<" ";}cout<<endl;}执行结果如下:。
循环赛日常表算法(N可为奇数和偶数)
循环赛⽇常表算法(N可为奇数和偶数)⼀、实验题⽬设有n位选⼿参加⽹球循环赛,循环赛共进⾏n-1天,每位选⼿要与其他n-1位选⼿⽐赛⼀场,且每位选⼿每天必须⽐赛⼀场,不能轮空。
试按此要求为⽐赛安排⽇程。
⼆、实验⽬的1.深刻理解并掌握“分治算法”的设计思想;2.提⾼应⽤“分治算法”设计技能;3.理解这样⼀个观点:⽤递归⽅法编写的问题解决程序具有结构清晰,可读性强等优点,且递归算法的设计⽐⾮递归算法的设计往往要容易⼀些,所以当问题本⾝是递归定义的,或者问题所涉及到的数据结构是递归定义的,或者是问题的解决⽅法是递归形式的时候,往往采⽤递归算法来解决。
三、实验要求1.实现《⽹球循环赛》问题的分治算法,并进⾏算法时间复杂性分析。
2.对实现的分治算法进⾏改进;3.对上述改进后算法进⾏时间复杂性分析,通过实验结果分析对⽐,得出⾃⼰的结论和总结。
四、实验过程1、算法⼀:#include<stdio.h>#define N 64void GameTable(int k,int a[][N]){//n=2^k(k>=1)个选⼿参加⽐赛,⼆维数组a表⽰⽇程安排,数组下标从1开始int n=2;//k=0,两个选⼿⽐赛⽇程可直接求得//求解两个选⼿⽐赛⽇程,得到左上⾓元素a[1][1]=1;a[1][2]=2;a[2][1]=2;a[2][2]=1;int i,j,t;for(t=1;t<k;t++)//迭代处理,依次处理2^2,....,2^k个选⼿⽐赛⽇程{int temp=n;n=n*2;//填左下⾓元素for(i=temp+1;i<=n;i++)for(j=1;j<=temp;j++)a[i][j]=a[i-temp][j]+temp;//左下⾓元素和左上⾓元素的对应关系//将左下⾓元素抄到右上⾓for(i=1;i<=temp;i++)for(j=temp+1;j<=n;j++)a[i][j]=a[i+temp][(j+temp)%n];//将左上⾓元素抄到右下⾓for(i=temp+1;i<=n;i++)for(j=temp+1;j<=n;j++)a[i][j]=a[i-temp][j-temp];}for(i=1;i<=n;i++)//显⽰⽇程表for(j=1;j<=n;j++){printf("- ",a[i][j]);if(j==n)printf("n");}}void main(){int a[N][N];int k;printf("输⼊选⼿的个数:(注意为2的平⽅)");scanf("%d",&k);GameTable(k,a);}2、结果验证当两个选⼿,即k=1时当4个选⼿时,即k=2当8个选⼿,即k=3当16个选⼿时,即k=16时间复杂度分析:迭代处理的循环体内部3个循环语句,每个循环语句都是⼀个嵌套的for循环,且它们的执⾏次数相同,基本语句是最内层循环体的赋值语句,即填写⽐赛⽇程表的元素。
循环赛名次表以及接下来赛程安排表
1
暂未定
2
循环淘汰赛各个小组名次排名
排名
A组
B组
C组
D组
第一名
动科学院
林学院
电气学院
机械学院
第二名
商学院
化工学院
计电学院
环境学院
第三名
农学院
材料学院
物理学院
生科院
第四名
轻工学院
公ቤተ መጻሕፍቲ ባይዱ学院
新闻学院
文学院
第五名
政治学院
法学院
土木学院
资冶学院
第六名
教育学院
外语学院
数信学院
赛程安排表
赛组
日期
时间
场地
场次
对战学院
八强进四强
2012.10.30
17:20——19:20
机械学院气排球场地
1
动科学院VS化工学院
2
林学院VS商学院
2012.10.31
17:20——19:20
机械学院气排球场地
1
电气学院VS环境学院
2
机械学院VS计电学院
半决赛
2012.11.1
17:20——19:20
暂未定
1
暂未定
2
决赛
2012.11.2
17:20——19:20
双循环赛的编排方法
(二)双循环赛的编排方法双循环赛比赛轮次表的排法与单循环相同,只要排出第一循环,第二循环可按表重复一次(表3),也可重新抽签另排位置。
第二循环的比赛如何进行,应在竞赛规程中明确规定。
双循环赛的轮次与场次,均为单循环的一倍。
表3 5个队参加双循环比赛轮次安排表(三)分组循环赛的编排方法分组循环通常分预赛和决赛两个阶段。
1.预赛阶段按规程规定将参赛队分为几个小组,各组参照单循环编排,排出小组比赛表,然后确定种子队的位置。
分组循环赛一般按分组数或分组数的2倍数确定种子,若种子数与组数相等,则将种子队分别安排在各小组的1号位置,如种子队为组数的2倍,应采用“蛇形”排列法,将种子队依次排列在各小组的1、2号位置上,非种子队也应抽签后定位。
现将分组单循环赛抽签和“蛇形”排列法介绍如下:(1)首先在联席会上协商确定种子队:种子队数一般等于组的组数。
如果分4个组进行比赛,应有4个种子队。
为了使比赛更合理,也可以多选出几个种子队,但必须是组数的倍数。
如分4个组进行比赛,可确定8个种子队。
第一号种子队与第八号种子队编为一组;第二号种子队与第七号种子队编为一组,依此类推。
(2)抽签方法:种子队先抽签,确定各种子队的组别,然后其他各队再抽签确定组别。
例如,20个队分为4组,除8个种子队外,其他12个队再抽签。
签号分为4组,每组有相同的3个签,由12个队抽签确定组别,然后再把各队按组别填入各组的比赛轮次表中。
另外一种分组方法为蛇形排列分组,即按上一届名次进行分组。
例如,有16个队分为4个组时,其排法如表4:表4 16个队分4组比赛安排表2.决赛阶段各队在预赛阶段分组单循环赛中的名次,将决定其进入决赛阶段比赛的位置。
在预赛阶段已经相遇过的队,比赛成绩依然有效,决赛阶段不再进行比赛。
其常用的比赛方法有,同名次赛、分段赛、交叉赛、录取名次赛等。
(1)同名次赛:就是将各小组预赛中相同名次编在一起进行比赛,如预赛时四个组的第一名编成一组进行单循环赛,决出1~4名,各小组的第二名编在一起决出5~8名。
利用分治法设计循环赛日程表
利用分治法设计循环赛日程表摘要:对于单循环赛的比赛日程安排问题,利用分治算法给出了可读性较好的设计,并分析了各种假设下的时间复杂度。
关键词:分治算法;复杂度;递归;循环赛引言任何一个可以用计算机求解的问题所需的计算时间都与其规模有关。
问题的规模越小,越容易求解,所需的计算时间也越少。
分治法是计算机科学中经常使用的一种算法。
设计思想是,将一个难以直接解决的大问题,分割成一些规模较小的相同问题,以便各个击破,分而治之。
1分治法应用条件及一般步骤1.1 分治法的应用条件1.1.1能将n个数据分解成k个不同子集合,且得到的k个子集合是可以独立求解的子问题,其中11&&odd(n/2)) copyodd(n);else copy(n);}void copyodd(int n){int m=n/2;for(int i=0;i=m){a[i][j]=b[i];a[m+i][j]=(b[i]+m)%n;}else a[m+i][j]=a[i][j]+m;}for(j=1;j<m;j++){a[i][m+j]=b[i+j];a[b[i+j]][m+j]=i;}}}分析算法的时间性能:当n/2为奇数时,基本语句的执行次数是:当n/2为偶数时,基本语句的执行次数是:综上,时间复杂度为O(4k )。
2.3 运行结果及分析当输入n为2时,输出:0 11 0在上面n=2时的日程表(二维表)中,左边第一列表示球队编号,第二列表示在某天碰到的对手球队的编号。
推广之,对于n行n列的二维日程表,那么,a[0][0],a[1][0],……,a[n-1][0]表示参加循环赛的n支球队的编号;a[0][1],a[0][2],……,a[0][n-1]表示球队a[0][0]在第1,2,……,n-1天碰到的对手球队编号。
a[i][j](i,j<n)表示编号为a[i][0]的球队在第j日遇到的对手球队的编号。
以下同。
循环赛表
A 1周赏赏2周鸣3陈俊杰4杨起积分计算名次1周赏赏
2周鸣
3陈俊杰
4杨起
B 1刘睿2栾立东3王浩4甘托积分计算名次1刘睿
2栾立东
3王浩
4甘托
C 1王达2谈友文3周斌积分计算名次
1王达
2谈友文
3周斌
D 1周文军2彭峰3刘文积分计算名次
1周文军
2彭峰
3刘文
E 1谭成军2陈滔滔3彭涛积分计算名次1谭成军
2陈滔滔
3彭涛
F 1刘姝婧2何鸿3段云飞积分计算名次1刘姝婧
2何鸿
3段云飞
G 1郑顺飘2卢冬艳3朱益平积分计算名次1郑顺飘
2卢冬艳
3朱益平
H 1李谋成2王亚峰3孙中林积分计算名次1李谋成
2王亚峰
3孙中林
第一轮:周赏赏VS杨起周鸣VS陈俊杰刘睿VS甘托栾立东VS王浩谈友文VS周斌王达(轮空)彭峰VS刘文周文军(轮空)
陈滔滔VS彭涛谭成军(轮空)何鸿VS段云飞刘姝婧(轮空)
卢冬艳VS朱益平郑顺飘(轮空)王亚峰VS孙中林李谋成(轮空)第二轮:周赏赏VS周鸣陈俊杰VS杨起刘睿VS栾立东王浩VS甘托王达VS谈友文周斌(轮空)周文军VS彭峰刘文(轮空)
谭成军VS陈滔滔彭涛(轮空)刘姝婧VS何鸿段云飞(轮空)
郑顺飘VS卢冬艳朱益平(轮空)李谋成VS王亚峰孙中林(轮空)
第三轮:周赏赏VS陈俊杰周鸣VS杨起刘睿VS王浩栾立东甘托王达VS周斌谈友文(轮空)周文军VS刘文彭峰(轮空)
谭成军VS彭涛陈滔滔(轮空)刘姝婧VS段云飞何鸿(轮空)
郑顺飘VS朱益平卢冬艳(轮空)李谋成VS孙中林王亚峰(轮空)。
5循环赛日程表问题
算法实验报告
循环赛日程表
设有n个运动员,设计一个满足以下要求的比赛日程表:
(1)每个选手必须与其他n-1个选手各赛一次;
(2)每个选手一天只能赛一次;
(3)当n为偶数时,比赛在一共进行n-1天。
当为奇数时,比赛在一共进行n天。
提示:
对于一般的正整数n,当n是奇数时,增设一个虚拟选手n+1,将问题转换为n是偶数的情形,当选手与虚拟选手比赛时,表示轮空。
因此只要关注n为偶数的情形即可处理。
当n/2为偶数时,与n=2k的情形类似,可用分治法求解。
当n/2为奇数时,递归返回的轮空的比赛要做进一步处理。
其中一种处理是在前n/2比赛中让轮空选手与下一个未参赛选手进行比赛。
Java_swt实现_分治法循环赛日程表_论文
附件二【学生用】《算法分析与设计》综合训练实习报告题目:利用分治思想设计循环赛日程表学号姓名专业班级指导教师实践日期目录一、综合训练目的与要求 (1)二、综合训练任务描述 (1)三、算法设计 (1)四、详细设计及说明 (3)五、调试与测试 (5)六、实习日志 (6)七、实习总结 (6)八、附录:核心代码清单 (6)一、综合训练目的与要求本综合训练是软件工程专业重要的实践性环节之一,是在学生学习完《算法分析》课程后进行的综合练习。
本课综合训练的目的和任务:1. 巩固和加深学生对算法分析课程基本知识的理解和掌握;2. 培养利用算法知识解决实际问题的能力;3. 掌握利用程序设计语言进行算法程序的开发、调试、测试的能力;4. 掌握书写算法设计说明文档的能力;5. 提高综合运用算法、程序设计语言、数据结构知识的能力。
二、综合训练任务描述假设有n=2k 个运动员要进行网球循环赛。
设计一个满足一下要求的比赛日程表:1. 每个选手必须与其他n-1个选手各赛一次2. 每个选手一天只能赛一次3. 循环赛一共进行n-1天利用Java语言开发一个界面,输入运动员的个数,输出比赛日程表。
对于输入运动员数目不满足n=2k时,弹出信息提示用户。
三、算法设计(1) 文字描述假设n位选手顺序编号为1,2,3……n,比赛的日程表是一个n行n-1列的表格。
i行j列的表格内容是第i号选手在第j天的比赛对手。
根据分而治之的原则,可从其中一半选手的比赛日程,导出全体n位选手的的日程,最终细分到只有两位选手的比赛日程出发(2)框图比赛人数n=2^k ,求出kNNYNYNYYNYint i=1i<=Ni++a[1][i]=ii++m=1s=1s<=k?N=N/2int t=1t<=N?int i=m+1i<=2*mj=m+1j<=m+1a[i][j+(t-1)*m*2] = a[i-m][j+(t-1)*m*2-m] a[i][j+(t-1)*m*2-m] = a[i-m][j+(t-1)*m*2]j++i++t++s++ m=m*2结束(3)伪代码public static void Table(int n,int[][]a) {int b = 2;n = saicheng.x;//参赛人数int k=(int) (Math.log(n)/Math.log(b)); //计算输入值是2的几次幂for(int i=1;i<=n;i++){a[1][i]=i;//打印出第一行即选手1的赛程表}int m=1;//控制每一次填充表格时i(i表示行)和j(j表示列)的起始填充位置for(int s=1;s<=k;s++){n/=2;//将问题分成k部分for(int t=1;t<=n;t++)//对每一部分进行划分{for(int i=m+1;i<=2*m;i++)//控制行{for(int j=m+1;j<=2*m;j++)//控制列{a[i][j +(t - 1) * m *2] = a[i - m][j + (t - 1) * m * 2 - m];//右下角等于左上角a[i][j + (t - 1) * m * 2 -m] = a[i - m][j + (t - 1) * m * 2];//左下角等于右上角}}}m*=2;}}四、详细设计及说明1.输入一个数字n,根据(x&(x-1))==0判断n是否等于2^k。
循环赛日程表(Java实现)
循环赛⽇程表(Java实现) 1/**2 * 循环赛⽇程表:有n = 2^k个运动员要进⾏⽹球循环赛3 * 赛程表满⾜:4 * 每个选⼿必须与其他n-1个选⼿各赛⼀次5 * 每个选⼿⼀天只能参赛⼀次6 * 循环赛在n-1天内结束7 *8 * 解题思路:9 * 将⽐赛⽇程表设计成⼀个n⾏和n-1列的表,第i⾏,第j列分别填⼊第i个选⼿在第j天所遇到的选⼿10 * 栗⼦:11 * 4个选⼿12 * ---------13 * |1|2|3|4|14 * ---------15 * |2|1|4|3|16 * ---------17 * |3|4|1|2|18 * ---------19 * |4|3|2|1|20 * ---------21 * 分治思想:将所有区域看成四块,区域1:(0,i) 区域2:(0,r+i) 区域3:(r,i) 区域4:(r,r+i)22 * 递归执⾏的是区域1拷贝到区域4,区域2拷贝到区域323 * ---------24 * | 1 | 2 |25 * ---------26 * | 3 | 4 |27 * ---------28 * * @author焦含寒29 *30*/31public class Roundrobin {32public static int[][] table(int k){33int n = 1<<k;34int[][] a = new int[n][n];35//构造赛程表第⼀⾏数据36for(int i = 0; i<n;i++)37 a[0][i] = i+1;38//采⽤分治算法,构造整个赛程表39for(int r = 1;r<n;r<<=1){40for(int i =0;i<n;i += 2*r){41 copy(a,r,r+i,0,i,r);42 copy(a,r,i,0,r+i,r);43 }44 }45return a;46 }4748private static void copy(int[][] a, int tox, int toy,49int fromx, int fromy, int r){50for(int i =0;i<r;i++){51for(int j = 0;j<r;j++){52 a[tox+i][toy+j] = a[fromx+i][fromy+j];53 }54 }5556 }57585960public static void main(String[] args) {6162int[][] a = table(4);63for(int i=0;i<a.length;i++){64for(int j = 0;j<a[0].length;j++){65 System.out.print(a[i][j] + "ss ");66 }67 System.out.println();68 }6970 }7172 }。
要求:编写程序,用分治法求解循环赛日程表。
要求:编写程序,用分治法求解循环赛日程表。
一、实验目的与要求1、掌握网球循环赛日程表的算法;2、初步掌握分治算法二、实验题:问题描述:有n=2^k个运动员要进行循环赛。
现要设计一个满足以下要求的比赛日程表:(1)每个选手必须与其他n-1个选手各赛一次(2)每个选手一天只能赛一次(3)循环赛一共进行n-1天三、实验代码#include <stdio.h>#include <stdlib.h>#define MAX 1024int a[MAX][MAX];void Copy(int tox, int toy, int fromx, int fromy, int n){ int i, j;for (i=0; i<n; i++){ for (j=0; j<n; j++){ a[tox + i][toy + j] = a[fromx + i][fromy + j];}}}void Table(int k, int a[][MAX]){ int i, n = 1 << k;for (i=0; i<n; i++){ a[0][i] = i + 1;}for (int r=1; r<n; r<<=1){ for (i=0; i<n; i+=2*r){ Copy(r, i + r, 0, i, r);Copy(r, i, 0, i + r, r);}}}void Out(int a[][MAX], int n){ int i, j;for (i=0; i<n; i++){ for (j=0; j<n; j++){ printf("%3d", a[i][j]);} printf("\n");} printf("\n");}int main(){ int i;for (i=0; i<5; i++){ int len = 1 << i;Table(i, a);Out(a, len);} return 0;}四、实验结果。
算法分析思维分析,以循环赛日程表为例
算法分析思维分析,以循环赛⽇程表为例第⼀步:分治法的简单思想在计算机科学中,分治法是⼀种很重要的算法。
字⾯上的解释是“分⽽治之”,就是把⼀个复杂的问题分成两个或更多的相同或相似的⼦问题,再把⼦问题分成更⼩的⼦问题……直到最后⼦问题可以简单的直接求解,原问题的解即⼦问题的解的合并。
这个技巧是很多⾼效算法的基础,如排序算法(,归并排序),傅⽴叶变换()等等。
任何⼀个可以⽤计算机求解的问题所需的计算时间都与其规模有关。
问题的规模越⼩,越容易直接求解,解题所需的计算时间也越少。
例如,对于n个元素的排序问题,当n=1时,不需任何计算。
n=2时,只要作⼀次⽐较即可排好序。
n=3时只要作3次⽐较即可,…。
⽽当n较⼤时,问题就不那么容易处理了。
要想直接解决⼀个规模较⼤的问题,有时是相当困难的。
分治法的设计思想是,将⼀个难以直接解决的⼤问题,分割成⼀些规模较⼩的相同问题,以便各个击破,分⽽治之。
分治策略是:对于⼀个规模为n的问题,若该问题可以容易地解决(⽐如说规模n较⼩)则直接解决,否则将其分解为k个规模较⼩的⼦问题,这些⼦问题互相独⽴且与原问题形式相同,递归地解这些⼦问题,然后将各⼦问题的解合并得到原问题的解。
这种算法设计策略叫做分治法。
第⼆步:分治法的理论基础如果原问题可分割成k个⼦问题,1<k≤n ,且这些⼦问题都可解并可利⽤这些⼦问题的解求出原问题的解,那么这种分治法就是可⾏的。
由分治法产⽣的⼦问题往往是原问题的较⼩模式,这就为使⽤递归技术提供了⽅便。
在这种情况下,反复应⽤分治⼿段,可以使⼦问题与原问题类型⼀致⽽其规模却不断缩⼩,最终使⼦问题缩⼩到很容易直接求出其解。
这⾃然导致递归过程的产⽣。
分治与递归像⼀对孪⽣兄弟,经常同时应⽤在算法设计之中,并由此产⽣许多⾼效算法。
2.1分治法所能解决的问题⼀般具有以下⼏个特征: 1) 该问题的规模缩⼩到⼀定的程度就可以容易地解决 2) 该问题可以分解为若⼲个规模较⼩的相同问题,即该问题具有最优⼦结构性质。
网球循环赛日程表
⽹球循环赛⽇程表⼀、问题表述:设有n个运动员要进⾏⽹球循环赛。
设计⼀个满⾜以下要求的⽐赛⽇程表,(1)每个选⼿必须与其他n-1个选⼿各赛⼀次;(2) 每个选⼿⼀天只能赛⼀次;(3) 当n是偶数时,循环赛进⾏n-1天,当n是奇数时,循环赛进⾏n天⼆、分析问题题⽬是要n名运动员进⾏循环⽐赛。
当n为偶数时,正好每天都可以两两⼀组,与其余的n-1个选⼿⽐赛,只需n-1天;⽽当n为奇数,每天将有⼀个选⼿轮空,⽐赛将持续n天。
可以采⽤的算法如下:1.算法⼀:使⽤分治法当n为偶数时,可以讲问题分为两个部分n/2; 然后继续划分,知道最后剩余两名选⼿单独⽐赛。
当n为奇数时,增设⼀个虚拟选⼿,运动员为n+1个,将问题转化为是偶数的情形。
当选⼿与虚拟选⼿⽐赛时,表⽰轮空,因此只需要关注n为偶数的情形。
a)当n/2为偶数时,与n = 2^k情形类此。
b)当n/2为奇数时,增设⼀个虚拟的选⼿,递归返回的将有轮空的选⼿,可以讲在前⾯n/2轮⽐赛的选⼿与后⾯n/2轮空的选⼿进⾏⽐赛。
2.算法⼆:利⽤边是奇数的正多边形。
特点:以多边形中的任意⼀个顶点画对称轴,其余偶数对顶点相互对称。
N名选⼿编号为1~n,将其画成⼀个正多边形。
a)所以当n为奇数时,第⼀天1号休息,其余以⼀号为对称轴,两两对称打⽐赛,第⼆天开始⼀次轮流休息,其余⼀休息的那个⼈编号为对称轴,两两⽐赛。
这样⽐赛可进⾏n天。
如图:b) 当n 为偶数时,取出编号最⼤的,其他的组成⼀个正多边形,n 号⼀次顺序与1,2,。
n-1号选⼿⽐赛,其他与a )相同。
如图所⽰:(图中是从0开始编号)三、理论分析算法及实现1.算法⼀:使⽤分治法a) 算法的思路:按分治策略,可以将所有的选⼿对分为两组(如果n是偶数,则直接分为n/2每组,如果n是奇数,则取(n+1)/2每组),n个选⼿的⽐赛⽇程表就可以通过为(n/2或(n+1)/2)个选⼿设计的⽐赛⽇程表来决定。
递归地⽤这种⼀分为⼆的策略对选⼿进⾏分割,直到只剩下2个选⼿时,⽐赛⽇程表的制定就变得很简单。
- 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
- 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
- 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。
//n/2为奇数
代码实现
❖ void copyodd(int n) // n/2为奇数的合并
{
int m=n/2;
for(int i=0;i<m;i++)
{
b[i]=m+i;
b[m+i]=b[i];
}
//未完
代码实现
for(i=0;i<m;i++){
for(int j=0;j<m+1;j++)
}
时间复杂度分析
1. N/2为奇数 T(n)=O(4k)
有2个循环结构 基本语句是循环体内的赋值语句 T(n)=2+(+2)=0(4k),
2.N/2为偶数T(n)=O(4k)
有2个循环结构 基本语句是循环体内的赋值语句 T(n)=3=0(4k),
调试运行
N=4 N=3
3
0
1
3
2
1
0
第输一天出要求<0,(n=4时) 第二天 1>=
0
1
A
1
0
<2, 3>= B<1
,
第一天
0>=
A
B
B
A
第三天
2
3
A
<3
3
2
,
A
B
B
A
2>=
B
输第一出天 要第二求天 (第三n天=4时)
0
1
2
3
1
0
3
2
2
3
0
1
3
2
1
0
AC BD
A可以决定B,B=A+n/2 C=编号较大数,每天循环轮转一次 D=编号较小数,每天循环轮转一次
{//由左上角小块的值算出相应的左下角小块
if(a[i][j]>=m)
{
a[i][j]=b[i];
a[m+i][j]=(b[i]+m)%n;
}
else a[m+i][j]=a[i][j]+m;
代码实现
//由左上角小块的值算出相应的右上角和右下角 小块的 for(j=1;j<m;j++) { a[i][m+j]=b[i+j]; a[b[i+j]][m+j]=i; } }
解题算法
递归
N=2
分治法步骤
分解
递归
求解
合并
将原问题分解 若干个 规模小
相互独立 形式相同
子问题
再继续分解 为更小的 子问题
容易被解决 则直接解
子问题的解 合并为
原问题的解
分治法步骤
divide-and-conquer(P) { if ( | P | <= n0) adhoc(P); //解决小规模的问题 divide P into smaller subinstances P1,P2,...,Pk;//分解问题 for (i=1,i<=k,i++) yi=divide-and-conquer(Pi); //递归的解各子问 题 return merge(y1,...,yk); //将各子问题的解合并 为原问题的解 }
输出要求
有n行和n-1列的表。在表中的第i行,第j列处填入第i个选 手在第j天所遇到的选手。其中1≤i≤n,1≤j≤n-1
输出要求(n=4时) 有n行和n-1列的表。在表中的第i行,第j列处填
入第i个选手在第j天所遇到的选手。其中1≤i≤n, 1≤j≤n-1
第一天 第二天 第三天
0
1
2
3
1
0
3
2
2
❖输入判断 分治法应用
当n小于或等于1时,没有比赛。 当n是偶数时,至少举行n-1轮比赛. 当n是奇数时,至少举行n轮比赛,这时每轮必有一支
球队轮空。
❖ 统一奇偶性
当n为奇数时,可以加入第n+1支球队(虚拟球队,实 际上不存在),并按n+1支球队参加比赛的情形安排
比赛日程 。
只需要考虑输入为偶数
if(odd(n)){tourna(n+1);return;} //n为奇
tourna(n/2);
//n为偶数,分治
makecopy(n);
//合并
}
代码实现
void makecopy(int n) //合并算法 { if((n/2)>1&&odd(n/2)) copyodd(n);
时
else copy(n); }
目录
1. 问 题 描 述 2. 解 题 思 想 3. 代 码 实 现 4.性 能 分 析 4. 调 试 运 行
❖ 题目
问题描述
设计一个满足以下要求的比赛日程表
❖ 具体要求
(1)每个选手必须与其他n-1个选手各赛一次
(2)每个选手一天只能赛一次 (3)循环赛一共进行n-1天
❖当n/2为分偶治数 法应用 n=偶
与n 情形类似,直接用分治法求解
❖ 当n/2为奇数
递归返回的轮空的比赛要作进一步处理 前n/2轮比赛中让轮空选手与下一个未参赛选手进行比
赛
代码实现
void tourna(int n)
//改进的分治赛算法
{
if(n==1){a[0][0]=0;return;}
N=8
12345678 21436587 34127856 43218765 56781234 65872143 78563412 87654321
题目分析
Байду номын сангаас
N为偶
N为奇
将所有选 手 对分成 为 两组
加入一个 虚拟队员
凡治解众如治题寡,思分数想是也-。-分----治孙子法兵法
N/2
N/4 分治法
N=5 N=2
N=7 N=8
N=2
N=3
N=4
N=7
N=8