问题

某乡村小学有六个年级,每个年级有一个班,共六个班。

周一到周五,每天上6节课,共计30节课。

开设的课程

一年级:语(9)数(9)书(2)体(2)美(2)音(2)德(2)班(1)安(1)

二年级:语(9)数(9)书(2)体(2)美(2)音(2)德(2)班(1)安(1)

三年级:语(8)数(8)英(4)体(2)美(2)音(2)德(2)班(1)安(1)

四年级:语(8)数(8)英(4)体(2)美(2)音(2)德(2)班(1)安(1)

五年级:语(8)数(8)英(4)体(2)美(2)音(2)德(2)班(1)安(1)

六年级:语(8)数(8)英(4)体(2)美(2)音(2)德(2)班(1)安(1)

要求
  • 各门课课时必须相符
  • 周一最后一节课班会,周五最后一节课安全教育是固定的。
  • 上午只能排语、数、英
  • 全校只有两位音乐老师
  • 三年级的数学不能排在周五上午第三节(三年级数学潘老师家里有事)

分析

将每一个(时间空间)点对,作为一个元素。这些元素都具有各自的[状态1,状态2,状态3,状态4,状态5,状态6,状态7,状态8,状态9]共9个状态

时间空间) --> 状态

一种状态对应一个课程。

对每一个元素,遍历相应的9种状态。

解的长度是固定的,30 * 6 的二维数组

套用子集树模板即可。

本问题只需要解决存在性。也就是说,只要找到一个符合要求的解就ok了。

此问题的本质就是,各(时,空)点对的取各自状态搭配的问题。

代码


'''排课问题''' # 作者:hhh5460
# 写于:2017年5月30日22时33分
# 声明:此算法的版权归本人所有 m = 30 # 一周课时数(时间)
n = 6 # 全校班级数(空间)
o = 30 * 6 # 元素个数,即(时, 空)点对的个数 # 6个班开始的课程(状态空间)
a = [['语','数','书','体','美','音','德','班','安'], # 一年级
['语','数','书','体','美','音','德','班','安'], # 二年级
['语','数','英','体','美','音','德','班','安'], # 三年级
['语','数','英','体','美','音','德','班','安'], # 四年级
['语','数','英','体','美','音','德','班','安'], # 五年级
['语','数','英','体','美','音','德','班','安']] # 六年级 # 课时数
b = [[9,9,2,2,2,2,2,1,1],
[9,9,2,2,2,2,2,1,1],
[8,8,4,2,2,2,2,1,1],
[8,8,4,2,2,2,2,1,1],
[8,8,4,2,2,2,2,1,1],
[8,8,4,2,2,2,2,1,1]] x = [[0 for _ in range(n)] for _ in range(m)] # 一个解,m*n 的二维数组 is_found = False # 结束所有递归标志!!!!! # 冲突检测
def conflict(t, s):
'''只考虑刚刚排的x[t][s]''' global m,n,o,a,b,x # 一、各门课课时数必须相符(纵向看)
# 1.前面已经排的课时不能超
if [r[s] for r in x[:t+1]].count(x[t][s]) > b[s][a[s].index(x[t][s])]: # 黑科技,不要眼花!
return True
# 2.后面剩的课时不能不够
if [r[s] for r in x[:t+1]].count(x[t][s]) + (m-t-1) < b[s][a[s].index(x[t][s])]:
return True # 二、周一最后一节课班会,周五最后一节课安全教育是固定的。
# 1.周一最后一节课班会
if x[t][s] == '班' and t != 5:
return True
# 2.周五最后一节课安全教育
if x[t][s] == '安' and t != 29:
return True # 三、上午只能排语、数、英
if t % 6 in [0,1,2] and x[t][s] not in ['语','数','英']:
return True # 四、只有两个音乐老师(横向看)
# 前面已经排的班级不能有3个及以上的班级同时上音乐课
if x[t][s] == '音' and x[t][:s+1].count('音') >= 3:
return True # 五、三年级的数学不能排在周五上午第三节(三年级数学潘老师家里有事)
if x[t][s] == '数' and t==5*n+3-1:
return True return False # 无冲突 # 套用子集树模板
def paike(t, s): # 到达(t,s)时空点对的位置
global m,n,o,a,b,x, is_found if is_found: return # 结束所有递归 if t == m: # 超出最尾的元素
#print(x)
show(x) # 美化版
is_found = True # 只需找一个
else:
for i in a[s]: # 遍历第s个班级的对应的所有状态,不同的班级状态不同
x[t][s] = i
if not conflict(t, s): # 剪枝
ns = [s + 1, 0][s + 1 == n] # 行扫描方式
nt = [t, t + 1][s + 1 == n]
paike(nt, ns) # 去往(nt, ns)时空点对 # 可视化一个解x
def show(x):
import pprint pprint.pprint(x[:6]) # 全校的周一课表
pprint.pprint(x[6:12]) # 全校的周二课表
pprint.pprint(x[12:18]) # 全校的周三课表
pprint.pprint(x[18:24]) # 全校的周四课表
pprint.pprint(x[24:]) # 全校的周五课表 # 测试
paike(0, 0) # 从时空点对(0,0)开始

效果图

正在运行中... ... 稍后奉上。

补:现在时间2017年6月1日7时40分,程序已运行34+时,还没有结果,囧!

之所以这么慢,其原因可能是递归太多了!

好吧,那只能等我以后将递归版本改为迭代版本,再运行看其结果如何了。

python 回溯法 子集树模板 系列 —— 6、排课问题的更多相关文章

  1. python 回溯法 子集树模板 系列 —— 18、马踏棋盘

    问题 将马放到国际象棋的8*8棋盘board上的某个方格中,马按走棋规则进行移动,走遍棋盘上的64个方格,要求每个方格进入且只进入一次,找出一种可行的方案. 分析 说明:这个图是5*5的棋盘. 图片来 ...

  2. python 回溯法 子集树模板 系列 —— 17、找零问题

    问题 有面额10元.5元.2元.1元的硬币,数量分别为3个.5个.7个.12个.现在需要给顾客找零16元,要求硬币的个数最少,应该如何找零?或者指出该问题无解. 分析 元素--状态空间分析大法:四种面 ...

  3. python 回溯法 子集树模板 系列 —— 16、爬楼梯

    问题 某楼梯有n层台阶,每步只能走1级台阶,或2级台阶.从下向上爬楼梯,有多少种爬法? 分析 这个问题之前用分治法解决过.但是,这里我要用回溯法子集树模板解决它. 祭出元素-状态空间分析大法:每一步是 ...

  4. python 回溯法 子集树模板 系列 —— 15、总结

    作者:hhh5460 时间:2017年6月3日 用回溯法子集树模板解决了这么多问题,这里总结一下使用回溯法子集树模板的步骤: 1.确定元素及其状态空间(精髓) 对每一个元素,遍历它的状态空间,其它的事 ...

  5. python 回溯法 子集树模板 系列 —— 14、最长公共子序列(LCS)

    问题 输入 第1行:字符串A 第2行:字符串B (A,B的长度 <= 1000) 输出 输出最长的子序列,如果有多个,随意输出1个. 输入示例 belong cnblogs 输出示例 blog ...

  6. python 回溯法 子集树模板 系列 —— 10、m着色问题

    问题 图的m-着色判定问题 给定无向连通图G和m种不同的颜色.用这些颜色为图G的各顶点着色,每个顶点着一种颜色,是否有一种着色法使G中任意相邻的2个顶点着不同颜色? 图的m-着色优化问题 若一个图最少 ...

  7. python 回溯法 子集树模板 系列 —— 9、旅行商问题(TSP)

    问题 旅行商问题(Traveling Salesman Problem,TSP)是旅行商要到若干个城市旅行,各城市之间的费用是已知的,为了节省费用,旅行商决定从所在城市出发,到每个城市旅行一次后返回初 ...

  8. python 回溯法 子集树模板 系列 —— 8、图的遍历

    问题 一个图: A --> B A --> C B --> C B --> D B --> E C --> A C --> D D --> C E -- ...

  9. python 回溯法 子集树模板 系列 —— 3、0-1背包问题

    问题 给定N个物品和一个背包.物品i的重量是Wi,其价值位Vi ,背包的容量为C.问应该如何选择装入背包的物品,使得放入背包的物品的总价值为最大? 分析 显然,放入背包的物品,是N个物品的所有子集的其 ...

  10. python 回溯法 子集树模板 系列 —— 13、最佳作业调度问题

    问题 给定 n 个作业,每一个作业都有两项子任务需要分别在两台机器上完成.每一个作业必须先由机器1 处理,然后由机器2处理. 试设计一个算法找出完成这n个任务的最佳调度,使其机器2完成各作业时间之和达 ...

随机推荐

  1. 机器学习实战(Machine Learning in Action)学习笔记————06.k-均值聚类算法(kMeans)学习笔记

    机器学习实战(Machine Learning in Action)学习笔记————06.k-均值聚类算法(kMeans)学习笔记 关键字:k-均值.kMeans.聚类.非监督学习作者:米仓山下时间: ...

  2. JavaScript获取请求参数

    <script type="text/javascript"> //获取请求参数 function paramsMap() { var url = window.loc ...

  3. js拼接table查询信息部分

    其一: $("#datagrid").empty(); var a=0; <order-rows> a++; $("#datagrid").appe ...

  4. MySQL优化之Explain命令解读,optimizer_trace

    简述: explain为mysql提供语句的执行计划信息.可以应用在select.delete.insert.update和place语句上.explain的执行计划,只是作为语句执行过程的一个参考, ...

  5. Huawei vlan 配置及vlan 间通讯

    Huawei Vlan配置及vlan 间通讯实例 组网需求:汇聚层交换机做为 PC 电脑的网关, PC3直连 SW2 属于 vlan 2,网关为 vlanif 2 接口地址192.168.2.1/24 ...

  6. 【转】Java学习---内存泄露与溢出的区别

    Java内存泄露与溢出的区别 Java内存泄漏就是没有及时清理内存垃圾,导致系统无法再给你提供内存资源(内存资源耗尽): 而Java内存溢出就是你要求分配的内存超出了系统能给你的,系统不能满足需求,于 ...

  7. 2018亚太CDN峰会开幕, 阿里云王海华解读云+端+AI的短视频最佳实践

    4月11-12日,2018亚太CDN峰会在北京隆重召开,在11日下午的短视频论坛中,阿里云高级技术专家王海华进行了<短视频最佳实践:云+端+AI>的主题演讲,分享了短视频的生命周期关键点和 ...

  8. mysql数据库管理工具(navicat for mysql) 10.1.7 绿色中文版

    Navicat for MySQL:Navicat for MySQL 是一套专为 MySQL 设计的高性能数据库管理及开发工具.它可以用于任何版本 3.21 或以上的 MySQL 数据库服务器,并支 ...

  9. Office 手动kms激活方法

    服务作用:在线激活windows和office 适用对象:VOL版本的windows和office 适用版本:截止到win10和office2016的所有版本 服务时间:24H,偶尔更新维护 优点:在 ...

  10. 4.Dubbo2.5.3集群容错和负载均衡

    转载请出自出处:http://www.cnblogs.com/hd3013779515/ 1.集群容错和负载均衡原理 各节点关系: 这里的Invoker是Provider的一个可调用Service的抽 ...