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