N皇后问题及深度优先算法.docVIP

  1. 1、原创力文档(book118)网站文档一经付费(服务费),不意味着购买了该文档的版权,仅供个人/单位学习、研究之用,不得用于商业用途,未经授权,严禁复制、发行、汇编、翻译或者网络传播等,侵权必究。。
  2. 2、本站所有内容均由合作方或网友上传,本站不对文档的完整性、权威性及其观点立场正确性做任何保证或承诺!文档内容仅供研究参考,付费前请自行鉴别。如您付费,意味着您自己接受本站规则且自行承担风险,本站不退款、不进行额外附加服务;查看《如何避免下载的几个坑》。如果您已付费下载过本站文档,您可以点击 这里二次下载
  3. 3、如文档侵犯商业秘密、侵犯著作权、侵犯人身权等,请点击“版权申诉”(推荐),也可以打举报电话:400-050-0827(电话支持时间:9:00-18:30)。
  4. 4、该文档为VIP文档,如果想要下载,成为VIP会员后,下载免费。
  5. 5、成为VIP后,下载本文档将扣除1次下载权益。下载后,不支持退款、换文档。如有疑问请联系我们
  6. 6、成为VIP后,您将拥有八大权益,权益包括:VIP文档下载权益、阅读免打扰、文档格式转换、高级专利检索、专属身份标志、高级客服、多端互通、版权登记。
  7. 7、VIP文档为合作方或网友上传,每下载1次, 网站将根据用户上传文档的质量评分、类型等,对文档贡献者给予高额补贴、流量扶持。如果你也想贡献VIP文档。上传文档
查看更多
N皇后问题及深度优先算法

常规N皇后解决问题过程 一.问题描述 运用回溯法解题通常包含以下三个步骤: (1)针对所给问题,定义问题的解空间; (2)确定易于搜索的解空间结构; (3)以深度优先的方式搜索解空间,并且在搜索过程中用剪枝函数避免无效搜索; ?通过上述的基本思路,我们可以将问题描述为:X(j)表示一个解的空间,j表示行数,里面的值表示可以放置在的列数,抽象约束条件得到能放置一个皇后的约束条件(1)X(i)!=X(k);(2)abs(X(i)-X(k))!=abs(i-k)。应用回溯法,当可以放置皇后时就继续到下一行,不行的话就返回到第一行,重新检验要放的列数,如此反复,直到将所有解解出。 也就是对于N×N的棋盘,选择出N个符合i!=rj!=s∧|i-r|!=|j-s|∨(i+r)!=(j+s)的点的排列总数。 ?二.伪代码: 判断点是否符合要求: ?place(k, X) ?? I=1 ?? While ik do ????? If x[i]==x[k] or abs(x[i]-x[k])==abs(i-k) then ???????? Return false ????? I=i+1 ????? Return true ?求问题的所有解: Nqueens(n, X) Sum=0 , X[1]=0 , k=1 While k0 do ?? X[k]=X[k]+1 ?? While X[k]=n and !(place(k,x)) ????? X[k]=X[k]+1 If X[k]=n then ?? Sum=Sum+1 Else ?? K=K+1 ,X[k]=0 Else ?? K=K-1 Print sum ?三.代码实现 #include iostream using namespace std; #include math.h ? /*检查可不可以放置一个新的皇后*/ bool place(int k, int *X) { ??? int i; ??? i=1; ??? while(ik) ??? { ? ???? if((X[i]==X[k])||(abs(X[i]-X[k])==abs(i-k))) ?? ??????? return false; ? ???? i++; ??? } ??? return true; } /*求解问题的所有解的总数,X存放列数*/ void Nqueens(int n,int *X) { ??? int k,sum=0; ??? X[1]=0; k=1; ??? while(k0) ??? { ? ???? X[k]=X[k]+1; ? ? ???? while((X[k]=n)(!place(k, X))) ??? ?????? X[k]=X[k]+1; ? ???? if(X[k]=n) ?? ??????? if(k==n) ?????????? { ????????????? for(int i=1;i=n;i++) ????????????????? coutX[i] ; ????????????? coutendl; ??? ????????? sum++; ?? ??????? } ?? ??????? else ?? ??????? { ??? ????????? k=k+1; ??? ????????? X[k]=0; ?? ??????? } ? ???????? else ????????????? k=k-1; ??? ?} ??? ?cout解的总数为:sumendl; } ? int main() { ??? int n; ??? int *X; ??? cout请输入皇后的个数:; ??? cinn; ?? X=new int[n]; ??? cout问题的解如下:endl; ??? Nqueens(n,X); ??? return 0;} 五.存在的问题 当皇后个数N大于等于16以上,程序对棋盘的扫描次数大到惊人: 从维基百科列出的结果不难看出,在25皇后时,符合条件的解集已经如此庞大了。而数组的存储及加法运算来求解已经不能适应当前的运算。 六.算法改进 程序中的所有数在计算机内存中都是以二进制的形式储存的,而位运算就是直接对整数在内存中的二进制位进行操作,所以速度快,效率高。因此我们选择用位运算来改进运算速度。 算法思想用图列应该更好解释: 如上图所示,假设一个8*8的棋盘,那么第一次我们在棋盘第一个位置放置一个皇后,则此时,第二列最靠右可放棋子的位置是3。假设第二个放到第二列3的位置,则此时,第三列最靠右能放棋子的位置是5...我们用蓝色线代表向右边斜的线,用橙色代表向左边斜的线,用红色代表向下边的线,而同一行,我们不需判断,因为棋子不能放置同一行的位置。这样

文档评论(0)

153****9595 + 关注
实名认证
文档贡献者

该用户很懒,什么也没介绍

1亿VIP精品文档

相关文档