草庐IT

八皇后问题(Python)

Vaeeeeeee 2023-05-10 原文

一.问题简介

八皇后问题: 如何能在 8*8 的国际象棋棋盘上放置八个皇后,使得任何一个皇后都无法直接吃掉其他的皇后?为了到达此目的,任两个皇后都不能处于同一条横行、纵行或斜线上。

二.几种思路和方法

1.回溯法+递归思想

 如图所示,圆圈代表皇后所放的位置,这里如果将棋盘转化为二维矩阵进行遍历比较麻烦,考虑到棋盘的每一行不能同时存在一个以上的皇后,所以将棋盘转化为一个具有八个元素的列表,而皇后的位置(i,j)对应的是列表中(元素的索引值,元素的值),因此放置皇后的操作变成了在列表中的每个位置填值操作,很明显的一个条件是列表中不能有相同的值。图中给出的是某一种情形

接下来直接看代码:

首先是定义一个queen函数,作用就是用来放皇后的位置。然后进入到第一个判断条件:如果当前行的位置遍历到“第9行”,即全部遍历完之后,将所有的结果放入到列表list中;然后是进入循环,在每一行的八个列的位置遍历,如果满足check中所有条件,那么直接递归调用,进入下一行遍历。

check函数:【任两个皇后都不能处于同一条横行】这一条件已经在前面保证成立了,这里只需保证剩下的两个条件【同一条纵行】和【同一条斜线】成立即可。注意:当两个皇后的位置坐标的连线斜率的绝对值为1,则这两个皇后在同一直线上。

def check(self, list, row):
        for i in range(row):
            if list[i] == list[row] or abs(list[row] - list[i]) == abs(row - i):
                return False
        return True
def queen(self, list, row, n):  # list为存储的结果;row是当前行;n为棋盘大小
        x=[]
        if row == n:
            x.extend(list)
            self.solves.append(x)
            return
        for i in range(n):  # 这里的i指的是所在列的值
            list[row] = i
            if self.check(list, row):
                self.queen(list, row + 1, n)

下面是完整代码:

导包:

import numpy as np           # 提供维度数组与矩阵运算
import copy                  # 从copy模块导入深度拷贝方法
from board import Chessboard

棋盘类的初始化

# 初始化8*8八皇后棋盘
chessboard = Chessboard()

# 基于棋盘类,设计搜索策略
class Game:
    def __init__(self, show = True):
        """
        初始化游戏状态.
        """
        
        self.chessBoard = Chessboard(show)
        self.solves = []
        self.gameInit()
        
    # 重置游戏
    def gameInit(self, show = True):
        """
        重置棋盘.
        """
        
        self.Queen_setRow = [-1] * 8
        self.chessBoard.boardInit(False)
                                                                        
    def check(self, list, row):
        for i in range(row):
            if list[i] == list[row] or abs(list[row] - list[i]) == abs(row - i):
                return False
        return True

    def queen(self, list, row, n):  # list为存储的结果;row是当前行;n为棋盘大小
        x=[]
        if row == n:
            x.extend(list)
            self.solves.append(x)
            return
        for i in range(n):  # 这里的i指的是所在列的值
            list[row] = i
            if self.check(list, row):
                self.queen(list, row + 1, n)
    
    def run(self, row=0):
        #self.solves.append([0, 6, 4, 7, 1, 3, 5, 2])
        n = 8
        list = [0 for _ in range(n)]
        self.queen(list,row,n)
        
    
    def showResults(self, result):
        """
        结果展示.
        """
        
        self.chessBoard.boardInit(False)
        for i,item in enumerate(result):
            if item >= 0:
                self.chessBoard.setQueen(i,item,False)
        
        self.chessBoard.printChessboard(False)
    
    def get_results(self):
        """
        输出结果.
        return: 八皇后的序列解的list.
        """
        
        self.run()
        return self.solves
   

验证一下结果:

game = Game()
solutions = game.get_results()
print('There are {} results.'.format(len(solutions)))
game.showResults(solutions[0])

 可以直接输出所有的结果(92种):

print(solutions)

2.排列组合的思想

在上面的基础上,可以稍微转化一下思路,放置不同的皇后位置就是将不同的八个元素的作排列组合问题,然后只要再保证“不在同一斜线”这个条件就可以了。下面是代码:

list1=[0,1,2,3,4,5,6,7]
result=[]
final_result=[]
from itertools import  permutations  #permutations函数:可以对集合或字符串进行排序或排列的所有可能的组合
for i in permutations(list1,8):      
    result.append(i)               #转化为排列组合问题,result存储部分可能的结果      

#下面只需要从result中找出满足‘任一两个皇后不在同一斜线上’的结果
for i in result:
    n=0                  #n用以标记那些满足条件的结果;
    for j in enumerate(i):
        for m in enumerate(i):
            if  m!=j and abs(m[0]-j[0])==abs(m[1]-j[1]):  #m!=j是避免对同一个坐标位置进行判断
                n+=1                                       #只要有两个坐标的连线斜率为1,n就不为0
    if n==0:
        final_result.append(i)                                
                
print(len(final_result))
print(final_result)

3.遗传算法

基本原理和步骤   

        遗传算法将要解决的问题模拟成一个生物进化的过程,通过复制、交叉、突变等操作产生下一代的解,并逐步淘汰掉适应度函数值低的解,增加适应度函数值高的解。这样进化N代后就很有可能会进化出适应度函数值很高的个体。

其计算步骤如下:

(1) 编码:将问题空间转换为遗传空间;

(2)创建初始种群:随机生成P 个染色体作为初始种群;

(3)适应度计算:染色体评价,计算各个染色体的适应度;

(4)选择操作:根据染色体适应度,按选择算子进行染色体的选择;

(5)交叉操作:按交叉概率进行交叉操作;

(6)变异操作:按变异概率进行变异操作;

(7)返回(4)形成新的种群,继续迭代,直到满足终止条件。

 

具体实现代码部分还在写,以后有机会在续上吧。

觉得不错的话,大家给个关注给个赞吧!

有关八皇后问题(Python)的更多相关文章

  1. python - 如何使用 Ruby 或 Python 创建一系列高音调和低音调的蜂鸣声? - 2

    关闭。这个问题是opinion-based.它目前不接受答案。想要改进这个问题?更新问题,以便editingthispost可以用事实和引用来回答它.关闭4年前。Improvethisquestion我想在固定时间创建一系列低音和高音调的哔哔声。例如:在150毫秒时发出高音调的蜂鸣声在151毫秒时发出低音调的蜂鸣声200毫秒时发出低音调的蜂鸣声250毫秒的高音调蜂鸣声有没有办法在Ruby或Python中做到这一点?我真的不在乎输出编码是什么(.wav、.mp3、.ogg等等),但我确实想创建一个输出文件。

  2. ruby - 在 64 位 Snow Leopard 上使用 rvm、postgres 9.0、ruby 1.9.2-p136 安装 pg gem 时出现问题 - 2

    我想为Heroku构建一个Rails3应用程序。他们使用Postgres作为他们的数据库,所以我通过MacPorts安装了postgres9.0。现在我需要一个postgresgem并且共识是出于性能原因你想要pggem。但是我对我得到的错误感到非常困惑当我尝试在rvm下通过geminstall安装pg时。我已经非常明确地指定了所有postgres目录的位置可以找到但仍然无法完成安装:$envARCHFLAGS='-archx86_64'geminstallpg--\--with-pg-config=/opt/local/var/db/postgresql90/defaultdb/po

  3. ruby - 通过 rvm 升级 ruby​​gems 的问题 - 2

    尝试通过RVM将RubyGems升级到版本1.8.10并出现此错误:$rvmrubygemslatestRemovingoldRubygemsfiles...Installingrubygems-1.8.10forruby-1.9.2-p180...ERROR:Errorrunning'GEM_PATH="/Users/foo/.rvm/gems/ruby-1.9.2-p180:/Users/foo/.rvm/gems/ruby-1.9.2-p180@global:/Users/foo/.rvm/gems/ruby-1.9.2-p180:/Users/foo/.rvm/gems/rub

  4. ruby - 通过 RVM (OSX Mountain Lion) 安装 Ruby 2.0.0-p247 时遇到问题 - 2

    我的最终目标是安装当前版本的RubyonRails。我在OSXMountainLion上运行。到目前为止,这是我的过程:已安装的RVM$\curl-Lhttps://get.rvm.io|bash-sstable检查已知(我假设已批准)安装$rvmlistknown我看到当前的稳定版本可用[ruby-]2.0.0[-p247]输入命令安装$rvminstall2.0.0-p247注意:我也试过这些安装命令$rvminstallruby-2.0.0-p247$rvminstallruby=2.0.0-p247我很快就无处可去了。结果:$rvminstall2.0.0-p247Search

  5. ruby - Fast-stemmer 安装问题 - 2

    由于fast-stemmer的问题,我很难安装我想要的任何ruby​​gem。我把我得到的错误放在下面。Buildingnativeextensions.Thiscouldtakeawhile...ERROR:Errorinstallingfast-stemmer:ERROR:Failedtobuildgemnativeextension./System/Library/Frameworks/Ruby.framework/Versions/2.0/usr/bin/rubyextconf.rbcreatingMakefilemake"DESTDIR="cleanmake"DESTDIR=

  6. ruby - 安装 Ruby 时遇到问题(无法下载资源 "readline--patch") - 2

    当我尝试安装Ruby时遇到此错误。我试过查看this和this但无济于事➜~brewinstallrubyWarning:YouareusingOSX10.12.Wedonotprovidesupportforthispre-releaseversion.Youmayencounterbuildfailuresorotherbreakages.Pleasecreatepull-requestsinsteadoffilingissues.==>Installingdependenciesforruby:readline,libyaml,makedepend==>Installingrub

  7. java - 从 JRuby 调用 Java 类的问题 - 2

    我正在尝试使用boilerpipe来自JRuby。我看过guide从JRuby调用Java,并成功地将它与另一个Java包一起使用,但无法弄清楚为什么同样的东西不能用于boilerpipe。我正在尝试基本上从JRuby中执行与此Java等效的操作:URLurl=newURL("http://www.example.com/some-location/index.html");Stringtext=ArticleExtractor.INSTANCE.getText(url);在JRuby中试过这个:require'java'url=java.net.URL.new("http://www

  8. ruby-on-rails - 简单的 Ruby on Rails 问题——如何将评论附加到用户和文章? - 2

    我意识到这可能是一个非常基本的问题,但我现在已经花了几天时间回过头来解决这个问题,但出于某种原因,Google就是没有帮助我。(我认为部分问题在于我是一个初学者,我不知道该问什么......)我也看过O'Reilly的RubyCookbook和RailsAPI,但我仍然停留在这个问题上.我找到了一些关于多态关系的信息,但它似乎不是我需要的(尽管如果我错了请告诉我)。我正在尝试调整MichaelHartl'stutorial创建一个包含用户、文章和评论的博客应用程序(不使用脚手架)。我希望评论既属于用户又属于文章。我的主要问题是:我不知道如何将当前文章的ID放入评论Controller。

  9. Python 相当于 Perl/Ruby ||= - 2

    这个问题在这里已经有了答案:关闭10年前。PossibleDuplicate:Pythonconditionalassignmentoperator对于这样一个简单的问题表示歉意,但是谷歌搜索||=并不是很有帮助;)Python中是否有与Ruby和Perl中的||=语句等效的语句?例如:foo="hey"foo||="what"#assignfooifit'sundefined#fooisstill"hey"bar||="yeah"#baris"yeah"另外,类似这样的东西的通用术语是什么?条件分配是我的第一个猜测,但Wikipediapage跟我想的不太一样。

  10. java - 什么相当于 ruby​​ 的 rack 或 python 的 Java wsgi? - 2

    什么是ruby​​的rack或python的Java的wsgi?还有一个路由库。 最佳答案 来自Python标准PEP333:Bycontrast,althoughJavahasjustasmanywebapplicationframeworksavailable,Java's"servlet"APImakesitpossibleforapplicationswrittenwithanyJavawebapplicationframeworktoruninanywebserverthatsupportstheservletAPI.ht

随机推荐