首页 > 其他分享 >回溯法求解n皇后问题(复习)

回溯法求解n皇后问题(复习)

时间:2022-12-12 16:12:17浏览次数:38  
标签:结点 xi 复习 求解 int 搜索 回溯 皇后

回溯法

回溯法是最常用的解题方法,有“通用的解题法”之称。当要解决的问题有若干可行解时,则可以在包含问题所有解的空间树中,按深度优先的策略,从根节点出发搜索解空间树。算法搜索至解空间树的任一结点时,总是先判断该结点是否肯定不包含问题的解。如果肯定不包含,则跳过对以该结点为根的子树的搜索,继续查找该结点的兄弟结点,若它的兄弟结点都不包含问题的解,则返回其父结点——这个步骤称为回溯。否则进入一个可能包含解的子树,继续按深度优先的策略进行搜索。这种以深度优先的方式搜索问题的解的算法称为回溯法。它本质上是一种穷举法,但由于在搜索过程中不断略过某些显然不合适的子树,所以搜索的空间大大少于一般的穷举,故它适用于解一些组合数较大的问题。

总结一下:

一、基本定义

回溯法(back track method)是在包含问题的所有可能解的解空间树中,从根结点出发,按照深度优先的策略进行搜索,对于解空间树的某个结点,若满足约束条件,则进入该子树继续搜索,否则将以该结点为根结点的子树进行剪枝。

二、适用范围

可避免搜索所有的可能解,适用于求解组合数较大的问题。

三、n皇后问题

问题:在n x n的棋盘上摆放n个皇后,而且n个皇后中的任意两个是不能处于同一行、同一列、或同一斜线上。

用数组x[i](1≤i≤n)表示n后问题的解。其中x[i]表示皇后i放在棋盘的第i行的第x[i]列。由于不允许将2个皇后放在同一列,所以解向量中的x[i]互不相同。2个皇后不能放在同一斜线上是问题的隐约束。对于一般的n后问题,这一隐约束条件可以化成显约束形式。设2个皇后放置位置为(i,j),(k,l):

显然,棋盘的每一行上可以而且必须摆放一个皇后,所以,n皇后问题的可能解用一个n元向量X=(x1, x2, …, xn)表示,其中,1≤i≤n并且1≤xi≤n,即第i个皇后放在第i行第xi列上

由于两个皇后不能位于同一列上,所以,解向量X必须满足约束条件:

xi≠xj (式1)

若两个皇后摆放的位置分别是(i, xi)和(j, xj),在棋盘上斜率为-1的斜线上,满足条件i-j= xi-xj,在棋盘上斜率为1的斜线上,满足条件i+j= xi+xj,综合两种情况,由于两个皇后不能位于同一斜线上,所以,解向量X必须满足约束条件:

|i-xi|≠|j-xj| (式2)

为了简化问题,下面讨论四皇后问题:

四皇后问题的解空间树是一个完全4叉树,树的根结点表示搜索的初始状态,对应Backtrack(1,x);从根结点到第2层结点对应皇后1在棋盘中第1行的可能摆放位置,从第2层结点到第3层结点对应皇后2在棋盘中第2行的可能摆放位置,依此类推。

完全4叉树,我只画了一部分,完整的应该是除了叶结点,每个内部结点都有四个子结点,k表示层数:

img

剪枝之后:

img

回溯法求解4皇后问题的搜索过程:

img

当然这个图只表示到找到的第一个解,我们知道还有另外一个解。

代码

变量sum记录可行方案个数,初始为0;

n表示皇后个数,由用户输入;

x[]数组保存问题的解,表示皇后i放在棋盘的第i行第x[i]列,初始时各元素都为0,而我们目的是求出有多少组(x[1],x[2],x[3]……x[n])满足摆放条件;

output(int x[])函数作用是输出当前找到的一个可行解,只在搜索到叶节点时才会调用;

Place(int k,int x[])函数作用是,对当前行k以上的所有行(即1到k-1行)逐行进行检查,如果该行与上面任何一行相互攻击(即位于同一对角线上了或同列了:abs(i-k)abs(x[i]-x[k]) || x[i]x[k]),那么返回false,否则返回true;

Backtrack(int k,int x[])函数表示搜索解空间中第k层子树,k>n时,算法搜索至叶节点,得到一个新的n皇后互不攻击放置方案,那么输出该方案,可行方案数sum加1;k<=n时,当前扩展节点是解空间的内部节点,该节点有x[1],x[2],x[3]……x[n]共n个子节点,对每一个子节点,用函数Place检查其可行性,如果可行,以深度优先的方式递归地对可行子树搜索,如果不可行剪枝。

#include <iostream>
#include <cmath>
using namespace std;
// 记录可行方案个数
int sum=0;
// 表示皇后个数,由用户输入
int n;
// 输出当前找到的一个可行解,只在搜索到叶节点时才会调用
int output(int  x[]){
    int i;
    for(i=1;i<=n;i++){
      cout << "(" << i << "," << x[i] << ")" << " ";                                   
    }
    cout << endl;
    return 0;
}

// 对当前行k以上的所有行(即1到k-1行)逐行进行检查
bool Place(int k,int  x[]){
     int i;
     for(i=1;i<k;i++){
     if(abs(i-k)==abs(x[i]-x[k]) || x[i]==x[k])
         return false;                     
     } 
     return true;
     
}
     
     
// 见上文详解     
int Backtrack(int k,int  x[]){
    int i;
     if(k>n){//如果是叶节点,直接输出找到的一个解 
          output(x);
          sum++; 
     }
     else{//内部节点,如果满足约束条件,继续深度搜索 。i代表列数,从1到n 
         for(i=1;i<=n;i++){
               x[k]=i;
               if(Place(k,x))
               Backtrack(k+1,x);
         } 
     }
     
     
}

int main(){
    int *x,i;
    
    cout << "输入皇后个数:" << endl;
    cin >> n;
    cout << endl;
    
    // 数组保存问题的解,表示皇后i放在棋盘的第i行第x[i]列
    x=new int[n+1];
    // 初始时各元素都为0
    for(i=0;i<=n;i++){
      x[i]=0;                
    }
    
    Backtrack(1,x);
    
    cout << endl;
    cout << "解的个数:" << sum << endl;
    
    system("pause"); 
    return 0;
}

运行结果

image

标签:结点,xi,复习,求解,int,搜索,回溯,皇后
From: https://www.cnblogs.com/galaxyStar/p/16976343.html

相关文章

  • 期末复习-数据库
    数据库基础知识一.单选题1-5DBCCD----------6-10DBBCA二.填空题1.文件系统中的数据独立性是指设备独立性2.文件系统的缺陷是:数据冗余、数据不一致和数据联系弱3.......
  • Control M 复习笔记
    记录一些复习过程想通的知识点1.我们教案中看到的图基本都是复平面,从来没有看到过所谓s域或z域,不同的稳定区域只是因为从复平面到函数中存在不同的映射过程(s函数和z函数)。......
  • 期末复习-计算机组成原理
    注:答案在后面,会更新大题的,填空题不更,因为不考,选择题下次也会慢慢更解析本科生期末试卷(一)一、选择题1从器件角度看,计算机经历了五代变化。但从系统结构看,至今绝大多数计......
  • 详解逻辑回归与评分卡-梯度下降求解逻辑回归【菜菜的sklearn课堂笔记】
    视频作者:菜菜TsaiTsai链接:【技术干货】菜菜的机器学习sklearn【全85集】Python进阶_哔哩哔哩_bilibili我们以最著名也最常用的梯度下降法为例。现在有一个带两个特征并......
  • Servlet+jsp复习
    ServletServlet是JavaEE十三项规范之一,是一个接口。javaweb的三大组件为Servlet、Filter、Listener。Servlet是一个运行在web服务器的java程序,可以用来接收客户端发来的......
  • 【面试高频题】难度 2/5,回溯算法经典运用
    题目描述这是LeetCode上的​​93.复原IP地址​​,难度为中等。Tag:「回溯」、「DFS」有效​​IP​​​地址正好由四个整数(每个整数位于​​0​​​到​​255......
  • 软件质量管理期末复习
    第一章软件质量和软件测试概述ISO/IEC25010中定义的软件产品质量模型包括下列的八个质量特性(掌握)功能性:特定条件下提供满足规定和功能需求的程度性能效率:一定条......
  • 【MOSMA】基于粘菌算法求解多目标优化问题附matlab代码
    ✅作者简介:热爱科研的Matlab仿真开发者,修心和技术同步精进,matlab项目合作可私信。......
  • 基于蜂虎狩猎 (BEH) 算法求解单目标优化问题附matlab代码
    ✅作者简介:热爱科研的Matlab仿真开发者,修心和技术同步精进,matlab项目合作可私信。......
  • JavaSE复习day5
    JavaSE复习day5胡家伟16.equals&toString&编写编译运行equals概念equals是在object类中的方法,在object中equals是用来检查两个参数是否引用的是同一个对象obj.equals......