设为首页 收藏本站
查看: 1466|回复: 0

[经验分享] 八皇后问题(python 生成器)

[复制链接]

尚未签到

发表于 2018-8-13 13:40:26 | 显示全部楼层 |阅读模式
问题:
  在8×8格的国际象棋上摆放八个皇后,使其不能互相***,即任意两个皇后都不能处于同一行、同一列或同一斜线上,问有多少种摆法。大致是下面这种样式:
   DSC0000.jpg
  思路:
  第一步:皇后位置存放问题
  用列表或元组表示。索引表示皇后所在的横行。列表的值表示 皇后的 竖列。
  那么皇后位置可表示为: L  i in range(8) 且 len(L) =8
  第二步:冲突问题。
  这种情况,没有考虑到冲突问题:
  同一行,由于用索引 表示,所以不会起冲突。
  同一行, L[n] 等于 L 值。也或是说 :皇后a. 列表值 - 皇后b.列表值 = 0
  斜行问题:
  斜行有两个方面考虑,一种是正斜45度,一种是反斜45度。
  相当于汉字中的撇捺。但不管那种情况。如果两个皇后是同斜行,必然是这样:
  | 皇后a.行值 - 皇后b.行值  | == | 皇后a.列值 - 皇后b. 列值  |
  绝对值【皇后a.索引-皇后b.索引 】= 绝对值 【皇后a.列表值 - 皇后b.列表值。】 。如下图所示:

设计处理模型:
  第一步:皇后摆放顺序 。伪代码为说明:(假设总的要摆放皇后的个数为num =8 )
  以上图为例,皇后按 行 一个一个摆放。
  当摆放第N+1个时,整个棋盘状态: next = N+1
  
  已摆放的位置放在列表 status 中: len ( status ) == N 。
  
  第 N+1 个,能摆放的位置 是   range( num )。 for pos in range( num )
  但是,这里需要排除 起冲突的。
# 第N+1 个皇后能摆放的位置:  

# 此时:皇后位置列表:status,已摆放:len(status)  

    for pos in range(num):if not drop_place(pos,status):    # 如果没有冲突继续摆放,否则返回。  

第二步:排除冲突。
  每一个已摆放好位置的皇后都要与第 N+1 个皇后 做比较,
  for i in range(N):   第 i  行。

  列值冲突  ,相等:  status  -- pos == 0

  斜着冲突: 行值 差等于 列值差的绝对值 。即
  | status – pos | =  len(status)  - i

#  pos 表示当前位置, status 表示前面摆放位置  
def drop_place(pos,status):
  

  nextY = len(status)        # 当前行号
  

  for i in range( nextY ):       # 已存在的位置都要比较
  if  abs(status - pos ) in (0, nextY - i ):     #如果有位置冲突
  return True           # 直接返回冲突,否则继续比较
  return False           # 最后一个比较完,没有冲突返回 False.注意缩进
  

第三步,是否继续摆放:
  这时需要考虑的是,本次摆放 的是不是 最后一个位置。
  如果不是,那继续摆放。(递归摆放)
  如果是最后一个位置.即 num –1. 如果是 那么是返回 pos 位置 还是 pos + status呢?
  和第一个代码结合 :

def queens(num=8, status=[]):for pos in range(num):if not drop_place(pos,status):# 下面是继续摆放  if len(status) != num -1:
  for each in queens(num,status+[pos,]): # 第一个?号
  yield [pos,]+each                   # 第二个?号
  

  else:        # 最后一个
  yield [pos,]                          # 第三个?号
  

  这里要解释下,为什么使用迭代生成器 而 不用 return。
  第N 个皇后摆放时,有 range(num) 个位置。如果,使用 return,那么当第一个位置满足条件时,直接返回。我们这里需要的是所有满足摆放的位置。
  位置是多个,所以 ,这里使用  for each in queens(num, status +[pos,]) . each 表示第 N+2 个皇后 满足 不冲突的位置。
  status + [pos,] 表示当 添加 N+2 个皇后时,此时队列必须要加上N+1的位置。
  第二个问号: 这里 为什么 用 生成 器 而不用 return ,就像我们上面说的那样,要生成所有满足 条件 的N+2位置,而不是一个位置就返回。
  再看返回的队列,[pos,] + each.
  这里 ,要再回想回想 递归的要求,必须是 递归的条件一步步满足停止递归的要求,否则递归 就是无限循环。
  这里,我们要摆放完所有的皇后,必须是基于最后一个皇后的位置存在,然后,倒着存入 所有的位置。
  而在摆放第N+2个皇后时,能确认的只有,pos + each 位置。
  当 each = 最后一个皇后时,就会从最后一个位置反着添加所有皇后的位置,从而生成整个符合条件的位置。
  第三个问号: 递归到最后一个皇后时,依然需要 使用 for each in queens(num, status+[pos,]) 得到最后一们的位置迭代对象。
  所以,这里使用 yield 返回 [pos,],再依次相加。最后,得到符合条件的一列数组。
  大致代码如下:
def drop_place(pos,status):  nextY
= len(status)for i in range( nextY ):if  abs(status - pos ) in (0, nextY - i ):return Truereturn False  

  

def queens(num=8, status=[] ):for pos in range(num):if not drop_place(pos,status):if len(status) == num - 1:yield [pos,]else:for result in queens(num, status +[pos,]):yield [pos,]+result  

  

print( len(list(queens(8))) )  

# 显示的长度为 92  

运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-551256-1-1.html 上篇帖子: python 获取yahoo股票数据 下篇帖子: python的基本数据类型:列表的方法
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表