所谓回溯算法,在笔者看来就是一种直接地思想----假设需要很多步操作才能求得最终的解,每一步操作又有很多种选择,那么我们就直接选择其中一种并依次深入下去。直到求得最终的结果,或是遇到明细的错误,回溯到上一步,换一种选择继续。就像把每种结果都遍历一遍,找到我们需要的结果。

  回溯算法非常适合使用递归来求解,但与一般的递归又稍有不同。一个递归需要递归公式+递归终止条件,当然使用递归来实现的回溯算法也需要这些,只是就笔者的理解而言回溯算法还需要“回溯”这一部分。所谓回溯,就是在某一步中有多个选择的时候,选择完某一个选择后可能造成一些影响,把这些影响消除后再去选择同级的另一个选择。(笔者自己的理解,可能有不准确或是更好地描述,还望不吝赐教)

  废话不多说,来看看十分著名的“N皇后”问题。题目来源与力扣(LeetCode),传送门

51.N皇后

  n 皇后问题研究的是如何将 n 个皇后放置在 n×n 的棋盘上,并且使皇后彼此之间不能相互攻击。

  给定一个整数 n,返回所有不同的 n 皇后问题的解决方案。

  每一种解法包含一个明确的 n 皇后问题的棋子放置方案,该方案中 'Q' 和 '.' 分别代表了皇后和空位。

  示例:

  输入: 4
  输出: [
  [".Q..", // 解法 1
  "...Q",
  "Q...",
  "..Q."],

  ["..Q.", // 解法 2
  "Q...",
  "...Q",
  ".Q.."]
  ]
  解释: 4 皇后问题存在两个不同的解法。

  

  这里稍微解释一下,N皇后问题最初来源与国际象棋的,而国际象棋的棋盘大小是8X8,因此经典版本的问题也叫“8皇后问题”。而所谓的皇后彼此之间不能相互攻击指的是:每个皇后棋子所在的行,所在的列,包括所在的对角线(两条)都没有其它棋子。

  解决N皇后问题的思路是这样的,依次将棋子放到第一行,第二行...第N行。每次放置的时候都检查一下当前放棋子的位置,检查是否满足“皇后彼此之间不能相互攻击”。如果不满足要求,就在同一行内换一个位置继续尝试。如果满足要求,就到下一行继续放置棋子。贴一下笔者的代码,抛砖引玉了。(由于输出结果格式的一些问题,添加了一些不必要的代码,可以优化掉)

 public class Solution {
private int count;
private IList<IList<string>> result = new List<IList<string>>();
IList<IList<string>> output = new List<IList<string>>(); public IList<IList<string>> SolveNQueens(int n)
{
count = n;
//初始化结果集合
for (int i = ; i < n; i++)
{
result.Add(new List<string>());
for (int j = ; j < n; j++)
{
result[i].Add(".");
}
}
//从第0行开始调用
calNQueens(); return output;
} private void calNQueens(int row)
{
if (row == count) //已经遍历了N行了,可以返回结果了,内层是关于格式的调整,可以优化掉的
{
List<string> temp = new List<string>(); for (int i = ; i < count; i++)
{
temp.Add(string.Join("", result[i]));
} output.Add(temp); return;
} //依次检查该列的每个位置
for (int column = ; column < count; column++)
{
if (isOk(row, column)) //放置棋子前检查是否满足要求,满足就放置棋子,并进入下一行。
{
result[row][column] = "Q";
calNQueens(row + );
}
//注意下面这一行,是笔者认为和一般的递归有些不同的地方,就是每次将“影响”消除,即“回溯”。对应到题目中去就是将刚刚放好的棋子再拿起来,避免出现一行有多个棋子的情况。
result[row][column] = ".";
} }
// 检查要放置棋子的位置是否满足条件
private bool isOk(int row, int column)
{
int leftup = column - ; //用于检查左上方的对角线
int rightup = column + ; //用于检查右上方的对角线
// 下面一段是用于检查同行内是否有重复的棋子,但其实这段逻辑是不需要的,因为在上面的调用过程中已经避免了这种情况。
// for (int i = count- ; i >= ; i--)
// {
// if (result[row][i].Equals("Q"))
// {
// return false;
// }
// } for (int i = row - ; i >= ; i--)
{
if (result[i][column].Equals("Q")) //检查同列
{
return false;
} if (leftup >= ) //检查左上方
{
if (result[i][leftup].Equals("Q"))
{
return false;
}
} if (rightup < count) //检查右上方
{
if (result[i][rightup].Equals("Q"))
{
return false;
}
} leftup--;
rightup++;
} return true;
}
}

  笔者自己的思路大致是上面的样子,更详细一点的解法可以参照官方给出的解答,传送门。除了回溯的想法以外,官方的解法还有一点很有意思的优化。利用到一个公式:

对于所有的主对角线(右上到左下)有 行号 + 列号 = 常数。对于所有的次对角线(左上到右下)有 行号 - 列号 = 常数。感兴趣的小伙伴可以自己去看一哈。

  关于回溯算法还有很多很多经典的问题,比如0-1背包等等,有机会再更新把。

算法刷题--回溯算法与N皇后的更多相关文章

  1. C#LeetCode刷题-回溯算法

    回溯算法篇 # 题名 刷题 通过率 难度 10 正则表达式匹配   18.8% 困难 17 电话号码的字母组合   43.8% 中等 22 括号生成   64.9% 中等 37 解数独   45.8% ...

  2. 8皇后以及N皇后算法探究,回溯算法的JAVA实现,非递归,循环控制及其优化

    上两篇博客 8皇后以及N皇后算法探究,回溯算法的JAVA实现,递归方案 8皇后以及N皇后算法探究,回溯算法的JAVA实现,非递归,数据结构“栈”实现 研究了递归方法实现回溯,解决N皇后问题,下面我们来 ...

  3. NOI题库分治算法刷题记录

    今天晚自习机房刷题,有一道题最终WA掉两组,极其不爽,晚上回家补完作业欣然搞定它,特意来写篇博文来记录下 (最想吐槽的是这个叫做分治的分类,里面的题目真的需要分治吗...) 先来说下分治法 分治法的设 ...

  4. 8皇后以及N皇后算法探究,回溯算法的JAVA实现,递归方案

    八皇后问题,是一个古老而著名的问题,是回溯算法的典型案例.该问题是国际西洋棋棋手马克斯·贝瑟尔于1848年提出:在8×8格的国际象棋上摆放八个皇后,使其不能互相攻击,即任意两个皇后都不能处于同一行.同 ...

  5. leetcode 算法刷题(一)

    今天开始刷Leetcode上面的算法题.我会更新我刷题过程中提交的代码(成功和不成功的都有)和比较好的解法 第二题 Add Two Numbers 题目的意思:输入两个链表,这两个链表都是倒序的数字, ...

  6. C#LeetCode刷题-贪心算法

    贪心算法篇 # 题名 刷题 通过率 难度 44 通配符匹配   17.8% 困难 45 跳跃游戏 II   25.5% 困难 55 跳跃游戏   30.6% 中等 122 买卖股票的最佳时机 II C ...

  7. C#LeetCode刷题-分治算法

    分治算法篇 # 题名 刷题 通过率 难度 4 两个排序数组的中位数 C#LeetCode刷题之#4-两个排序数组的中位数(Median of Two Sorted Arrays)-该题未达最优解 30 ...

  8. leetcode算法刷题(三)

    今天在刷了几道简单的动态规划后,又看了看string方面的题 第五题 Longest Palindromic Substring 题目的意思:求一个字符串的最长回文子串 分析:开始,我的想法是,现在字 ...

  9. leetcode算法刷题(四)——动态规划(二)

    又到了晚上,动态规划,开刷! 第121题 Best Time to Buy and Sell Stock 题目的意思:给予一个数组price,表示特定股票在某天的股价,里面第i个数表示第i天的价格.只 ...

随机推荐

  1. lnmp1.5一键安装包安装lnmpa后,添加站点

    lnmp1.5一键安装包安装lnmpa后,添加站点 (1)添加站点 (2)配置apache配置文件 在/usr/local/apache/conf/vhost文件夹下,修改webApp站点配置文件ap ...

  2. 3ds Max File Format (Part 4: The first useful data; Scene, AppData, Animatable)

    The most interesting part of this file is, evidently, the Scene. Opening it up in the chunk parser, ...

  3. python接口自动化测试之根据excel中的期望结果是否存在于请求返回的响应值中来判断用例是否执行成功

    1.首先在excel中填写好预期结果的值 这里判断接口成功的依据是预期结果值是否存在于接口的返回数据中. 一般接口的返回值都是json对象,我们需要将json对象转换为json格式的字符串 如下图,进 ...

  4. 剑指offer系列——59/60.按之字形顺序打印二叉树/把二叉树打印成多行

    Q:请实现一个函数按照之字形打印二叉树,即第一行按照从左到右的顺序打印,第二层按照从右至左的顺序打印,第三行按照从左到右的顺序打印,其他行以此类推. A:BFS,偶数层reverse vector&l ...

  5. 在PDB级别中如何切换或重建UNDO表空间

    Oracle 12.1版本中,UNDO表空间仅存在CDB级别(共享UNDO),来自于AskScuti博客园. Oracle 12.2版本开始,UNDO表空间同时可以存在每个PDB级别(本地UNDO). ...

  6. \r、\n、\r\n的区别-转载

    文章地址: https://blog.csdn.net/qq_40395278/article/details/81199281 https://blog.csdn.net/qq592304796/a ...

  7. leetcode全部滑动窗口题目总结C++写法(完结)

    3. 无重复字符的最长子串 A: 要找最长的无重复子串,所以用一个map保存出现过的字符,并且维持一个窗口,用le和ri指针标识.ri为当前要遍历的字符,如果ri字符在map中出现过,那么将le字符从 ...

  8. 在Vue中输入框自动获取焦点的三种方式

    原生JS操作DOM使用mounted钩子函数,它表示页面一加载进来就执行函数里面的内容(和window.onload类似)1//html部分 编号:<input type="text& ...

  9. SpringBoot整合WEB开发--(四)@ControllerAdvice

    1.全局异常处理: @ControllerAdvice处理全局数据,一般搭配@ExceptionHandler,@ModelAttribute以及@InitBinder使用. @ControllerA ...

  10. js -- 正则表达式集合

    在做项目中,有时需要进行正则验证,但我又不太会正则表达式. 在一次又一次的寻找正则表达式的过程中,我最后总结了一个用于验证的函数,把我们常用的正则写在方法里,就不用每次都去网上找了. 可以根据需求进行 ...