当前位置: 首页 > news >正文

分书问题的递归枚举算法

分数问题的递归枚举算法

  • 一、问题引入
  • 二、解题步骤
    • 1.问题分析思维导图
    • 2.解题步骤
  • 三、代码实现
    • 1.代码
    • 2.复杂度分析
  • 四、个人总结

一、问题引入

分书问题是指:已知 n 个人对 m 本书的喜好(n≤m),现要将 m 本书分给 n 个人,每个人只能分到 1 本书,每本书也最多只能分给 1 个人,并且还要求每个人都能分到自己喜欢的书。列出所有满足要求的方案。

本题请你对任意 n 和 m 尝试列出全部的解。

输入格式:
输入第一行给出两个正整数 n 和 m(n≤m≤8),即分书问题中的人数和书的数量。
随后 n 行,每行给出 m 个关系矩阵元素。其中第 i 行第 j 个元素为 1 表示第 i 个人喜欢第 j 本书,为 0 则表示不喜欢。

输出格式:
按升序列出所有满足要求的方案,格式为 (s1,⋯,sn)。其中 si表示第 i 个人分到了第 si本书。

注:方案 (a1,⋯,an)<(b1,⋯,bn) 是指存在 1≤k≤n,使得 ai=bi对所有 1≤i<k 成立,并且有 ak<bk。

输入样例:

4 5
0 1 0 0 1
1 1 0 1 0
1 0 1 1 0
0 0 0 1 1

输出样例:

(2, 1, 3, 4)
(2, 1, 3, 5)
(2, 1, 4, 5)
(2, 4, 1, 5)
(2, 4, 3, 5)
(5, 1, 3, 4)
(5, 2, 1, 4)
(5, 2, 3, 4)

二、解题步骤

1.问题分析思维导图

在这里插入图片描述

2.解题步骤

  1. 问题理解:
    ◦ 有n个人和m本书,n≤m
    ◦ 每人只能分到一本自己喜欢的书
    ◦ 每本书只能分配给一个人
    ◦ 需要找出所有满足条件的分配方案
  2. 算法选择:
    ◦ 使用回溯算法递归枚举所有可能的分配方案
    ◦ 通过剪枝(只考虑喜欢的书和未被分配的书)减少不必要的搜索
  3. 实现步骤:
    ◦ 回溯函数:
    1.终止条件:当所有人都分配了书时,保存当前方案
    2.对于当前人,遍历所有书:
    ■ 如果书未被分配且当前人喜欢这本书:
    ■ 标记书为已分配
    ■ 递归处理下一个人
    ■ 回溯:取消书的分配状态
    ◦ 主函数:
    1.读取输入:n, m和喜好矩阵
    2.调用回溯函数获取所有方案
    3.对结果排序并输出

三、代码实现

1.代码

def solve_book_assignment(n, m, preference):def backtrack(person, assigned_books, current_assignment, results):if person == n:# 找到一个完整的分配方案results.append(tuple(current_assignment))returnfor book in range(m):if not assigned_books[book] and preference[person][book]:# 如果书未被分配且当前人喜欢这本书assigned_books[book] = Truecurrent_assignment[person] = book + 1  # 书的编号从 1 开始backtrack(person + 1, assigned_books, current_assignment, results)assigned_books[book] = False  # 回溯results = []assigned_books = [False] * m  # 记录书是否被分配current_assignment = [0] * n  # 记录当前分配方案backtrack(0, assigned_books, current_assignment, results)return resultsdef main():n, m = map(int, input().split())preference = []for _ in range(n):row = list(map(int, input().split()))preference.append(row)# 获取所有分配方案results = solve_book_assignment(n, m, preference)# 按升序输出for res in sorted(results):print(f"({', '.join(map(str, res))})")if __name__ == "__main__":main()

2.复杂度分析

时间复杂度:O(m!/(m-n)!)
■ 最坏情况下需要枚举所有排列,即从m本书中选n本的排列数
空间复杂度:O(n+m)

四、个人总结

通过本次实验,我深入理解了回溯算法在解决组合优化问题中的应用。实验以分书问题为载体,让我亲身体验了如何将实际问题转化为递归搜索模型。在实现过程中,我学会了如何设计递归函数的终止条件和回溯逻辑,特别是掌握了通过标记数组来避免重复选择的关键技巧。当遇到输出结果顺序不符合要求的情况时,我通过分析比较规则,理解了字典序的本质,最终采用预排序方案解决了输出顺序问题。

调试过程中出现的重复分配错误让我意识到状态恢复的重要性,这加深了我对回溯算法中"尝试-撤销"这一核心思想的理解。通过分析算法复杂度,我认识到虽然回溯法能保证找到所有解,但其时间代价随问题规模呈阶乘级增长,这让我更加重视剪枝优化的重要性。实验数据也验证了理论分析,当n和m接近8时,运行时间明显增加。

这次实践不仅让我掌握了回溯算法的实现细节,更重要的是培养了我将抽象算法转化为具体代码的能力。我体会到良好的数据结构设计对算法效率的影响,比如使用布尔数组而非列表来记录分配状态可以提升访问效率。

http://www.xdnf.cn/news/376147.html

相关文章:

  • [思维模式-25]:《本质思考力》-6- 马哲的三大规律:对立统一规律、质量互变规律、否定之否定规律,以及在计算机领域中的体现
  • RHCE实验:远程控制qq邮箱发送邮件
  • 20250510解决NanoPi NEO core开发板在Ubuntu core22.04.3系统下适配移远的4G模块EC200A-CN的问题
  • C++内存管理
  • 仓库管理系统,Java+Vue,含源码及文档,高效管理仓库物资,实现入库、存储、出库全流程数字化精准管控
  • 基于CNN卷积神经网络的带频偏QPSK调制信号检测识别算法matlab仿真
  • MySQL 从入门到精通(五):索引深度解析 —— 性能优化的核心武器
  • idea如何快速生成测试类
  • 【赵渝强老师】TiDB SQL层的工作机制
  • Yocto中`${B}`变量的作用
  • 论文图表自动编号与交叉引用
  • python中的继承和多态
  • FreeRTOS Queue消息队列-笔记
  • AlimaLinux设置静态IP
  • 护网HVV初级蓝队面试题总结
  • Axure :基于中继器的列表删除 、 列表编辑
  • 自动语音拨号系统V2.6.0产品说明书
  • Dockers部署oscarfonts/geoserver镜像的Geoserver
  • BERT类模型
  • CenOS7切换使用界面
  • 推荐一款免费开源工程项目管理系统软件,根据工程项目全过程管理流程开发的OA 办公系统
  • 基于定制开发开源AI智能名片S2B2C商城小程序的公私域流量融合运营策略研究
  • 策略路由更改路径
  • Best Video下载器——抖音视频去水印工具
  • day21python打卡
  • 【Linux第三章】vim
  • HTTP/2概览及内核解析
  • excel大表导入数据库
  • comfyu BiRefNet-General模型下载及存放地方
  • JS正则表达式介绍(JavaScript正则表达式)