循环日程赛问题
循环日程赛是利用分治法解决的经典问题之一。
问题描述
设有n = 2k 个选手参加比赛,要求设计一个日程赛安排表,满足:
- 每个选手都与其他n-1个选手各自比赛一次;
- 每个选手每天只比赛一次;
- 比赛进行n-1天。
设计分析
设计日程赛安排表可以用一个n行n-1列的二维数组来表示,a[i][j]即表示第i个选手在第j天遇到的对手。
要想利用分治法解决该问题,就必须将整个问题进行划分,划分成容易解决的子问题进行求解。按照分治法的思想,我们可以将n个选手一分为二,计算n/2个选手日程赛的情况,继续划分,直至两个选手时,此时为最简单的情况,以此解决n个选手的情况。
当n=2时,2个选手,此情况下最为简单:
1 | 2 |
2 | 1 |
当n=4时,4个选手,我们先用穷举的方法得到结果:
1 | 2 | 3 | 4 |
2 | 1 | 4 | 3 |
3 | 4 | 1 | 2 |
4 | 3 | 2 | 1 |
我们可以看出填表的规律:
- 表的左上子表等于表的右下子表,左下子表等于右上子表;
- 左下子表的元素等于左上角子表对应元素加2k-1;
- 左上子表对应的是前2k-1个选手的比赛前半程的比赛日程;左下角子表是后2k-1个选手的比赛前半程比赛日程。
根据这个规律我们再来验证n=8的情况:
1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 |
2 | 1 | 4 | 3 | 6 | 5 | 8 | 7 |
3 | 4 | 1 | 2 | 7 | 8 | 5 | 6 |
4 | 3 | 2 | 1 | 8 | 7 | 6 | 5 |
5 | 6 | 7 | 8 | 1 | 2 | 3 | 4 |
6 | 5 | 8 | 7 | 2 | 1 | 4 | 3 |
7 | 8 | 5 | 6 | 3 | 4 | 1 | 2 |
8 | 7 | 6 | 5 | 4 | 3 | 2 | 1 |
可以看出此规律是正确的。
因此在编写程序进行填表时我们就通过分治法的思想,进行划分,首先填写20个选手的情况,再根据规律填写21、22……2n的情况。
源代码
#include<iostream>
#include<vector>
using namespace std;
//非递归
void Arrange(vector<vector<int> > &a) {
if