002-3算法笔记【分治】-习题2-15球赛日程表

#include "iostream"
#include "cmath"
using namespace std;
//---------------------------------
//多边形法
//---------------------------------
int arr[20][20];
int b[20];

int odd(int n) //如果是奇数,返回1,否则返回0
{
    return n & 1;
}

void construct(int n)
{
    if (n == 1)
        return;
    int m = odd(n) ? n : n - 1; //如果n是奇数,转换为偶数
    arr[n][1] = n;
    for (int i = 1; i <= m; i++)
    {
        arr[i][1] = i; //第1列,表示第i个选手
        b[i] = b[i + m] = i + 1;
    }
    for (int i = 1; i <= m; i++) //第2列~第n列,相当于旋转多边形,确定哪个选手和哪个选手比赛
    {
        arr[1][i + 1] = b[i]; //每列中的第一个数字
        arr[b[i]][i + 1] = 1;
        for (int j = 1; j <= m / 2; j++) //每列中的其余数字
        {
            arr[b[i + m - j]][i + 1] = b[i + j]; //相当于水平线连起来的两个顶点
            arr[b[i + j]][i + 1] = b[i + m - j];
        }
    }
}

//---------------------------------
//分治法:根据结果反推出来的
//---------------------------------
int a[20][20];
void copy(int n);
void tourna(int n){
    if(n==1){
        a[1][1]=1;
        return;
    }
    tourna(n/2);
    //将左上角递归计算出的小块中的所有数字按其相对位置抄到右下角,
    //将右上角小块中的所有数字加 n/2 后按其相对位置抄到左下角,
    //将左上角的小块抄到右下角
    copy(n);
}
/*
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
*/
void copy(int n){
    int m=n/2;
    for(int i=1;i<=m;i++){
        for(int j=1;j<=m;j++){
            a[i+m][j+m] = a[i][j];//将左上角的小块抄到右下角
            //将左上角的小块加 n/2 后抄到右上角
            a[i][j+m] = a[i][j]+m;
            //将左下角小块中的所有数按其相对位置抄到右上角
            a[i+m][j] = a[i][j+m];
        }
    }
}

int main()
{
    int n;
    // cout << "输入参加比赛的运动员数量:";
    // cin >> n;
    n = 8;
    construct(n);

    cout << "多边形法,循环赛日程表为:\n";
    for (int i = 1; i <= n; i++)
    {
        for (int j = 1; j <= n; j++)
            cout << arr[i][j] << "\t";
        cout << endl;
    }

    tourna(n);
    cout << "分治法,循环赛日程表为:\n";
    for (int i = 1; i <= n; i++)
    {
        for (int j = 1; j <= n; j++)
            cout << a[i][j] << "\t";
        cout << endl;
    }
    return 0;
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
循环赛日程表问题是一个经典的组合数学问题,它的目标是设计一个比赛日程表,使得n个队伍两两之间都要比赛一次,且每个队伍每天只能进行一次比赛。这个问题可以使用分治算法解决。 下面是循环赛日程表分治算法的C语言代码实现: ```c #include<stdio.h> void Schedule(int **a, int n, int k) { if (n == 2) { a = 1; a = 2; a = 2; a = 1; } else { Schedule(a, n / 2, k); int i, j; for (i = 1; i <= n / 2; i++) { for (j = k + 1; j <= k + n / 2; j++) { a[i][j] = a[i + n / 2][j - n / 2]; } } for (i = n / 2 + 1; i <= n; i++) { for (j = k + 1; j <= k + n / 2; j++) { a[i][j] = a[i - n / 2][j - n / 2]; } } for (i = 1; i <= n / 2; i++) { for (j = k + n / 2 + 1; j <= k + n; j++) { a[i][j] = a[i + n / 2][j - n / 2 - 1] + n / 2; } } for (i = n / 2 + 1; i <= n; i++) { for (j = k + n / 2 + 1; j <= k + n; j++) { a[i][j] = a[i - n / 2][j - n / 2 - 1] + n / 2; } } Schedule(a, n / 2, k + n / 2); } } int main() { int n, i, j; printf("请输入参赛队伍数量:"); scanf("%d", &n); int **a = (int **) malloc((n + 1) * sizeof(int *)); for (i = 1; i <= n; i++) { a[i] = (int *) malloc((n + 1) * sizeof(int)); } Schedule(a, n, 0); printf("比赛日程表如下:\n"); for (i = 1; i <= n; i++) { for (j = 1; j <= n; j++) { printf("%d ", a[i][j]); } printf("\n"); } return 0; } ``` 该算法的思路是先把问题规模缩小到n/2,然后将n/2个队伍分成两组,第一组与第二组之间的比赛日程表可以由上一步得到,第一组与第三组之间、第二组与第四组之间的比赛日程表可以通过对上一步得到的日程表进行调整得到。然后再把问题规模缩小到n/4,重复上述过程,直到问题规模缩小到2。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值