仅利用30行Python代码来展示X算法


Posted in Python onApril 01, 2015

假如你对数独解法感兴趣,你可能听说过精确覆盖问题。给定全集 X 和 X 的子集的集合 Y ,存在一个 Y 的子集 Y*,使得 Y* 构成 X 的一种分割。

这儿有个Python写的例子。
 

X = {1, 2, 3, 4, 5, 6, 7}
Y = {
  'A': [1, 4, 7],
  'B': [1, 4],
  'C': [4, 5, 7],
  'D': [3, 5, 6],
  'E': [2, 3, 6, 7],
  'F': [2, 7]}

这个例子的唯一解是['B', 'D', 'F']。

精确覆盖问题是NP完备(译注:指没有任何一个够快的方法可以在合理的时间内,意即多项式时间 找到答案)。X算法是由大牛高德纳发明并实现。他提出了一种高效的实现技术叫舞蹈链,使用双向链表来表示该问题的矩阵。

然而,舞蹈链实现起来可能相当繁琐,并且不易写地正确。接下来就是展示Python奇迹的时刻了!有天我决定用Python来编写X 算法,并且我想出了一个有趣的舞蹈链变种。
算法

主要的思路是使用字典来代替双向链表来表示矩阵。我们已经有了 Y。从它那我们能快速的访问每行的列元素。现在我们还需要生成行的反向表,换句话说就是能从列中快速访问行元素。为实现这个目的,我们把X转换为字典。在上述的例子中,它应该写为
 

X = {
  1: {'A', 'B'},
  2: {'E', 'F'},
  3: {'D', 'E'},
  4: {'A', 'B', 'C'},
  5: {'C', 'D'},
  6: {'D', 'E'},
  7: {'A', 'C', 'E', 'F'}}

眼尖的读者能注意到这跟Y的表示有轻微的不同。事实上,我们需要能快速删除和添加行到每列,这就是为什么我们使用集合。另一方面,高德纳没有提到这点,实际上整个算法中所有行是保持不变的。

以下是算法的代码。
 

def solve(X, Y, solution=[]):
  if not X:
    yield list(solution)
  else:
    c = min(X, key=lambda c: len(X[c]))
    for r in list(X[c]):
      solution.append(r)
      cols = select(X, Y, r)
      for s in solve(X, Y, solution):
        yield s
      deselect(X, Y, r, cols)
      solution.pop()
 
def select(X, Y, r):
  cols = []
  for j in Y[r]:
    for i in X[j]:
      for k in Y[i]:
        if k != j:
          X[k].remove(i)
    cols.append(X.pop(j))
  return cols
 
def deselect(X, Y, r, cols):
  for j in reversed(Y[r]):
    X[j] = cols.pop()
    for i in X[j]:
      for k in Y[i]:
        if k != j:
          X[k].add(i)

真的只有 30 行!
格式化输入

在解决实际问题前,我们需要将输入转换为上面描述的格式。可以这样简单处理

X = {j: set(filter(lambda i: j in Y[i], Y)) for j in X}

但这样太慢了。假如设 X 大小为 m,Y 的大小为 n,则迭代次数为 m*n。在这例子中的数独格子大小为 N,那需要 N^5 次。我们有更好的办法。
 

X = {j: set() for j in X}
for i in Y:
  for j in Y[i]:
    X[j].add(i)

这还是 O(m*n) 的复杂度,但是是最坏情况。平均情况下它的性能会好很多,因为它不需要遍历所有的空格位。在数独的例子中,矩阵中每行恰好有 4 个条目,无论大小,因此它有N^3的复杂度。
优点

  •     简单: 不需要构造复杂的数据结构,所有用到的结构Python都有提供。
  •     可读性: 上述第一个例子是直接从Wikipedia上的范例直接转录下来的!
  •     灵活性: 可以很简单得扩展来解决数独。

求解数独

我们需要做的就是把数独描述成精确覆盖问题。这里有完整的数独解法代码,它能处理任意大小,3×3,5×5,即使是2×3,所有代码少于100行,并包含doctest!(感谢Winfried Plappert 和 David Goodger的评论和建议)

Python 相关文章推荐
Python对象的深拷贝和浅拷贝详解
Aug 25 Python
跟老齐学Python之编写类之四再论继承
Oct 11 Python
python3实现ftp服务功能(客户端)
Mar 24 Python
从0开始的Python学习014面向对象编程(推荐)
Apr 02 Python
Python中的 is 和 == 以及字符串驻留机制详解
Jun 28 Python
pandas按行按列遍历Dataframe的几种方式
Oct 23 Python
Pycharm小白级简单使用教程
Jan 08 Python
python读取图片的几种方式及图像宽和高的存储顺序
Feb 11 Python
Python网络爬虫信息提取mooc代码实例
Mar 06 Python
Python装饰器结合递归原理解析
Jul 02 Python
pytorch DataLoader的num_workers参数与设置大小详解
May 28 Python
Python集合set()使用的方法详解
Mar 18 Python
探究数组排序提升Python程序的循环的运行效率的原因
Apr 01 #Python
用Python编写分析Python程序性能的工具的教程
Apr 01 #Python
对Python新手编程过程中如何规避一些常见问题的建议
Apr 01 #Python
利用Django框架中select_related和prefetch_related函数对数据库查询优化
Apr 01 #Python
用实例详解Python中的Django框架中prefetch_related()函数对数据库查询的优化
Apr 01 #Python
Python的Django框架中的select_related函数对QuerySet 查询的优化
Apr 01 #Python
简单的Python2.7编程初学经验总结
Apr 01 #Python
You might like
php专用数组排序类ArraySortUtil用法实例
2015/04/03 PHP
PHP defined()函数的使用图文详解
2019/07/20 PHP
找到一点可怜的关于dojo资料,谢谢作者!
2006/12/06 Javascript
复制js对象方法(详解)
2013/07/08 Javascript
js实现简单登录功能的实例代码
2013/11/09 Javascript
JavaScript中的常见问题解决方法(乱码,IE缓存,代理)
2013/11/28 Javascript
一步步教大家编写酷炫的导航栏js+css实现
2016/03/14 Javascript
使用jQuery判断浏览器滚动条位置的方法
2016/05/30 Javascript
详解微信小程序——自定义圆形进度条
2016/12/29 Javascript
基于jquery实现多选下拉列表
2017/08/02 jQuery
Node.JS使用Sequelize操作MySQL的示例代码
2017/10/09 Javascript
React教程之封装一个Portal可复用组件的方法
2018/01/02 Javascript
JS实现获取毫秒值及转换成年月日时分秒的方法
2018/08/15 Javascript
Vue项目History模式404问题解决方法
2018/10/31 Javascript
Vue管理系统前端之组件拆分封装详解
2020/08/23 Javascript
解决Vue项目中tff报错的问题
2020/10/21 Javascript
[02:41]DOTA2英雄基础教程 谜团
2013/12/10 DOTA
[02:27]DOTA2英雄基础教程 莱恩
2014/01/17 DOTA
[00:16]热血竞技场
2019/03/06 DOTA
[40:03]RNG vs VG 2019国际邀请赛小组赛 BO2 第二场 8.15
2019/08/17 DOTA
python字典序问题实例
2014/09/26 Python
python中将函数赋值给变量时需要注意的一些问题
2017/08/18 Python
python使用matplotlib画柱状图、散点图
2019/03/18 Python
Python Tkinter 简单登录界面的实现
2019/06/14 Python
python aiohttp的使用详解
2019/06/20 Python
Charlotte Tilbury英国官网:英国彩妆品牌
2017/05/26 全球购物
Currentbody美国/加拿大:美容仪专家
2020/03/09 全球购物
俄罗斯连接商品和买家的在线平台:goods.ru
2020/11/30 全球购物
房地产还款计划书
2014/01/10 职场文书
高中生学期学习自我评价
2014/02/24 职场文书
计划生育证明格式范本
2014/09/12 职场文书
党的群众路线教育实践活动个人批评与自我批评
2014/10/16 职场文书
烈士陵园观后感
2015/06/08 职场文书
售房协议书范本
2015/08/11 职场文书
threejs太阳光与阴影效果实例代码
2022/04/05 Javascript
浅谈Redis的事件驱动模型
2022/05/30 Redis