运动员最佳配对问题(分支界限法)

2023-11-06 00:40

本文主要是介绍运动员最佳配对问题(分支界限法),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

运动员配对问题

一、问题描述

【问题简述】
羽毛球队有男女运动员各n人。给定2个n×n矩阵P和Q。P[i][j]是男运动员i和女运动员j配对组成混合双打的男运动员竞赛优势;Q[i][j]是女运动员i和男运动员j配合的女运动员竞赛优势。由于技术配合和心理状态等各种因素影响,P[i][j]不一定等于Q[j][i]。男运动员i和女运动员j配对组成混合双打的男女双方竞赛优势为P[i][j]×Q[j][i]。设计一个算法,计算男女运动员最佳配对法,使各组男女双方竞赛优势的总和达到最大。
【输入形式】
由文件input.txt给出输入数据。第1行有1个正整数n(1≤n≤20)。接下来的2n行,每行n个数。前n行是p,后n行是q。
【输出形式】
将计算的男女双方竞赛优势的总和的最大值输出到文件output.txt。

二、问题分析

【题目分析】
这道题目共有n!种配对情况,也就是相当于固定男运动员,然后对女运动员进行一次全排列,并求出对应的优势之和的最大值,本题可以用回溯法,也可以用分支限界法,在使用分支限界法的时候,关键是在于设计上界函数。
在这里,我们把上界函数定义为:剩下的未配对的女运动员(不考虑男运动员配对情况下)所能达到的优势最大值之和(记为r)与当前配对已达到的优势(记为sum)之和。
在使用分支限界法的时候,一旦搜索到所搜索过的叶节点,那么就立即结束算法,因为最先出来的叶节点必定是最优解。

三、算法设计

利用分支界限法求解此题:
①创建一个最大值堆,用于表示活结点优先队列
②队中每个结点的sum值是优先队列的优先级
③算法计算出每个顶点的最大sum值
④搜索到所搜索的排列数的叶子节点,算法结束,输出最大值

四、关键代码

【初始化数组】

file.open("input.txt",ios::in);		//打开输入文件file >> n;for (int i = 0; i < n; i++){for (int j = 0; j < n; j++){file >> P[i][j];}}for (int i = 0; i < n; i++){for (int j = 0; j < n; j++){file >> Q[i][j];}}file.close();	//关闭输入文件 

​ 利用两个二维数组来储存Pij和Qij,时间复杂度为O(n²),空间复杂度为O(n²)
【分支界限法求解】

while (Node->id != n ){for (int i = Node->id; i < n; i++){node* nNode = new node();nNode->id = Node->id + 1;nNode->x = new int[n];for (int t = 0; t < n; t++){nNode->x[t] = Node->x[t];}nNode->x[Node->id] = Node->x[i];nNode->x[i] = Node->x[Node->id];nNode->sum = Node->sum + P[Node->id][nNode->x[Node->id]] * Q[nNode->x[Node->id]][Node->id];nNode->r = Node->r - maxn[Node->id];nNode->up = nNode->sum + nNode->r;q.push(nNode);}if (!q.empty()){Node = q.top();q.pop();}else {tmp = 0;break;}}tmp = Node->sum;

​ 相较于回溯法的时间复杂度O(n!),分支界限法对排列树进行广度优先搜索,算法复杂度有所降低。

五、实验源码

#include<iostream>
#include<algorithm>
#include<queue>
#include <fstream>
using namespace std;
int P[20][20];
int Q[20][20];
int maxn[20];
int n;
fstream file;
struct node {int id;int sum;int r;int up;int* x;
};
struct cmp {bool operator()(node* a, node* b){return(a->up < b->up);}
};
int main()
{file.open("input.txt",ios::in);		//打开输入文件file >> n;for (int i = 0; i < n; i++){for (int j = 0; j < n; j++){file >> P[i][j];}}for (int i = 0; i < n; i++){for (int j = 0; j < n; j++){file >> Q[i][j];}}file.close();	//关闭输入文件 for (int i = 0; i < n; i++){int maxnum = 0;for (int j = 0; j < n; j++){maxnum = max(maxnum, Q[i][j] * P[j][i]);}maxn[i] = maxnum;}int tmp = 0;priority_queue<node*, vector<node*>, cmp>  q;node* Node = new node();Node->id = 0;Node->sum = 0;Node->r = 0;Node->up = 0;for (int i = 0; i < n; i++){Node->r += maxn[i];}Node->up = Node->r;Node->x = new int[n];for (int i = 0; i < n; i++){Node->x[i] = i;}while (Node->id != n ){for (int i = Node->id; i < n; i++){node* nNode = new node();nNode->id = Node->id + 1;nNode->x = new int[n];for (int t = 0; t < n; t++){nNode->x[t] = Node->x[t];}nNode->x[Node->id] = Node->x[i];nNode->x[i] = Node->x[Node->id];nNode->sum = Node->sum + P[Node->id][nNode->x[Node->id]] * Q[nNode->x[Node->id]][Node->id];nNode->r = Node->r - maxn[Node->id];nNode->up = nNode->sum + nNode->r;q.push(nNode);}if (!q.empty()){Node = q.top();q.pop();}else {tmp = 0;break;}}tmp = Node->sum;file.open("output.txt",ios::out); 	//打开输出文件  file << tmp;file.close();return(0);
}

实例输入txt文件:
在这里插入图片描述

六、实验心得

【回溯法与分支界限法的比较】
1.回溯法:
回溯法在问题的解空间树中,按深度优先策略,从根结点出发搜索解空间树。算法搜索至解空间树的任意一点时,先判断该结点是否包含问题的解。如果肯定不包含,则跳过对该结点为根的子树的搜索,逐层向其祖先结点回溯;否则,进入该子树,继续按深度优先策略搜索。这种以深度优先方式系统搜索问题解的算法称为回溯法。
2.分支限界法:
分支限界法是以广度优先或以最小耗费优先的方式搜索解空间树,在每一个活结点处,计算一个函数值,并根据函数值,从当前活结点表中选择一个最有利的结点作为扩展结点,使搜索朝着解空间上有最优解的分支推进,以便尽快地找出一个最优解,这种方法称为分支限界法。
3.回溯法的基本思想:
用回溯法解问题时,应明确定义问题的解空间。问题的解空间至少应包含问题的一个解。之后还应将解空间很好的组织起来,使得能用回溯法方便的搜索整个解空间。在组织解空间时常用到两种典型的解空间树,即子集树和排列树。确定了解空间的组织结构后,回溯法从开始结点出发,以深度优先方式搜索整个解空间。这个开始结点成为活结点,同时也成为当前的扩展结点。在当前的扩展结点处,搜索向纵深方向移至一个新结点。这个新结点就成为新的活结点,并成为当前扩展结点。如果在当前的扩展结点处不能再向纵深方向移动,则当前扩展结点就成为死结点。此时,应往回移动至最近的一个活结点处,并使这个活结点成为当前的扩展结点。回溯法以这种工作方式递归的在解空间中搜索,直至找到所要求的解或解空间中已无活结点时为止。
4.分支限界法的基本思想:
分支限界法常以广度优先或以最小耗费有限的方式搜索问题的解空间树。问题的解空间树是表示问题解空间的一棵有序树,常见的有子集树和排列树。在搜索问题的解空间树时,分支限界法和回溯法的主要区别在于它们对当前扩展节点所采用的扩展方式不同。在分支限界法中,每一个活结点只有一次机会成为扩展节点。活结点一旦成为扩展节点,就一次性产生其所有儿子节点。在这些儿子节点中,导致不可行解或导致非最优解的儿子节点被舍弃,其余儿子节点被加入活结点表中。此后,从活结点表中取下一节点为当前扩展节点。并重复上述节点扩展过程。这个过程移至持续到找到所需的解或活结点表为空为止。
从活结点表中选择下一扩展节点的不同方式导致不同的分支限界法。最常见的有以下两种方式:
(1)队列式分支限界法:队列式分支限界法将活结点表组织成一个队列,并按队列的先进先出原则选取下一个节点为当前扩展节点。
(2)有限队列式分支限界法:优先队列式的分支限界法将活结点表组织成一个优先队列,并按优先队列中规定的节点优先级选取优先级最高的下一个节点成为当前扩展节点。

【实验心得】
对排列树在分支界限法中的运用有了更好的理解与运用,并且对队列式分支限界法有了更加深刻的理解,也对分支限界法与回溯法的运用有了更好的区别运用。

下一个节点成为当前扩展节点。

【实验心得】
对排列树在分支界限法中的运用有了更好的理解与运用,并且对队列式分支限界法有了更加深刻的理解,也对分支限界法与回溯法的运用有了更好的区别运用。

这篇关于运动员最佳配对问题(分支界限法)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/353479

相关文章

关于@MapperScan和@ComponentScan的使用问题

《关于@MapperScan和@ComponentScan的使用问题》文章介绍了在使用`@MapperScan`和`@ComponentScan`时可能会遇到的包扫描冲突问题,并提供了解决方法,同时,... 目录@MapperScan和@ComponentScan的使用问题报错如下原因解决办法课外拓展总结@

MybatisGenerator文件生成不出对应文件的问题

《MybatisGenerator文件生成不出对应文件的问题》本文介绍了使用MybatisGenerator生成文件时遇到的问题及解决方法,主要步骤包括检查目标表是否存在、是否能连接到数据库、配置生成... 目录MyBATisGenerator 文件生成不出对应文件先在项目结构里引入“targetProje

C#使用HttpClient进行Post请求出现超时问题的解决及优化

《C#使用HttpClient进行Post请求出现超时问题的解决及优化》最近我的控制台程序发现有时候总是出现请求超时等问题,通常好几分钟最多只有3-4个请求,在使用apipost发现并发10个5分钟也... 目录优化结论单例HttpClient连接池耗尽和并发并发异步最终优化后优化结论我直接上优化结论吧,

Java内存泄漏问题的排查、优化与最佳实践

《Java内存泄漏问题的排查、优化与最佳实践》在Java开发中,内存泄漏是一个常见且令人头疼的问题,内存泄漏指的是程序在运行过程中,已经不再使用的对象没有被及时释放,从而导致内存占用不断增加,最终... 目录引言1. 什么是内存泄漏?常见的内存泄漏情况2. 如何排查 Java 中的内存泄漏?2.1 使用 J

numpy求解线性代数相关问题

《numpy求解线性代数相关问题》本文主要介绍了numpy求解线性代数相关问题,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 在numpy中有numpy.array类型和numpy.mat类型,前者是数组类型,后者是矩阵类型。数组

解决systemctl reload nginx重启Nginx服务报错:Job for nginx.service invalid问题

《解决systemctlreloadnginx重启Nginx服务报错:Jobfornginx.serviceinvalid问题》文章描述了通过`systemctlstatusnginx.se... 目录systemctl reload nginx重启Nginx服务报错:Job for nginx.javas

Redis缓存问题与缓存更新机制详解

《Redis缓存问题与缓存更新机制详解》本文主要介绍了缓存问题及其解决方案,包括缓存穿透、缓存击穿、缓存雪崩等问题的成因以及相应的预防和解决方法,同时,还详细探讨了缓存更新机制,包括不同情况下的缓存更... 目录一、缓存问题1.1 缓存穿透1.1.1 问题来源1.1.2 解决方案1.2 缓存击穿1.2.1

vue解决子组件样式覆盖问题scoped deep

《vue解决子组件样式覆盖问题scopeddeep》文章主要介绍了在Vue项目中处理全局样式和局部样式的方法,包括使用scoped属性和深度选择器(/deep/)来覆盖子组件的样式,作者建议所有组件... 目录前言scoped分析deep分析使用总结所有组件必须加scoped父组件覆盖子组件使用deep前言

解决Cron定时任务中Pytest脚本无法发送邮件的问题

《解决Cron定时任务中Pytest脚本无法发送邮件的问题》文章探讨解决在Cron定时任务中运行Pytest脚本时邮件发送失败的问题,先优化环境变量,再检查Pytest邮件配置,接着配置文件确保SMT... 目录引言1. 环境变量优化:确保Cron任务可以正确执行解决方案:1.1. 创建一个脚本1.2. 修

Python 标准库time时间的访问和转换问题小结

《Python标准库time时间的访问和转换问题小结》time模块为Python提供了处理时间和日期的多种功能,适用于多种与时间相关的场景,包括获取当前时间、格式化时间、暂停程序执行、计算程序运行时... 目录模块介绍使用场景主要类主要函数 - time()- sleep()- localtime()- g