八皇后问题应该是回溯法的教学典范。在本科的时候,有一门课叫面向对象。最后的附录有这个问题的源码。当时根本不懂编程,照抄下来,执行一下出了结果都非常开心,哎。

皇后们的限制条件是不能同行同列,也不能同对角线。

那么显然每一列上都要有一个皇后,仅仅须要用一个一维数组记录皇后在每一行上的位置就能够了。

算法的思想是:从第一行開始,尝试把皇后放到某一列上,能够用一个vis数组保存已经有皇后的列,当找到一个还没有皇后的列时,就尝试着把当前皇后放上,然后看看有没有之前放好的皇后跟这个皇后同对角线,假设同对角线的话,就仅仅能尝试后面的位置。

所有位置都尝试完成之后。说明在前面放的皇后的位置不正确。就要进入传说中的回溯环节,回溯过程中,行是要退回到上一行这是没有疑问的。关键是这一行应该从哪个位置開始继续尝试呢,用递归的话当然不用操心,他会从进入递归的下一位置開始,循环的话呢?应该从当前分配给他的下一个位置開始。同一时候。他之前所在的那一列应该释放掉。

循环的实现还有个问题,什么时候推出?假设仅仅求一组解的话好说,找到解,也就是所有的皇后都归位之后。就退出。可是生成所有解的话。在得到一组解之后,还要做回溯。我一開始对这样的时候的回溯想错了,想着如今应该让第一行的皇后移到下一个位置了,结果少了非常多解。

非常可能前面的几个皇后是不用更改,仅仅调换一下后面的几个就能够了,因此这样的情况的回溯跟尝试失败的回溯是全然一样的。

再来说退出条件。想一下回溯到最后会如何。会到达第一行。第一行将皇后移动到当前放置位置的下一个位置,因此当第一行尝试了所有的列,即下一列就超出棋盘的时候,就应该停止了。

class Solution {
public:
vector<vector<string> > solveNQueens(int n) {
int pos[n];
vector<vector<string> > res;
if(n == 0) return res;
vector<string> tpres;
memset(pos, 0, sizeof(pos));
int i=0, j=0, start=0;
bool flag, vis[n];
memset(vis, 0, sizeof(vis));
string line(n, '.'), tpline(n, '.');
while(i<n){
for(j=start;j<n;j++){
if(vis[j]) continue;
flag = true;
for(int k=0;k<i;k++){
if(abs(k-i) == abs(pos[k] - j)){
flag = false;
break;
}
}
if(!flag) continue;
pos[i] = j;
vis[j] = 1;
break;
}
if(j==n){
--i;
vis[pos[i]] = 0;
start = pos[i]+1;
if(i==0&&start>=n) break;
continue;
}else{
i++;
start = 0;
}
if(i == n){
for(i=0;i<n;i++){
line = tpline;
line[pos[i]] = 'Q';
tpres.push_back(line);
}
res.push_back(tpres);
tpres.clear();
--i;
vis[pos[i]] = 0;
start = pos[i]+1;
if(i==0&&start>=n) break;
}
}
return res;
}
};

leetcode第一刷_N-Queens的更多相关文章

  1. leetcode第一刷_Set Matrix Zeroes

    这个题乍一看非常easy,实际上还挺有技巧的.我最開始的想法是找一个特殊值标记.遇到一个0,把他所相应的行列中非零的元素标记成这个特殊值.0值保持不变,然后再从头遍历一次,碰到特殊值就转化成0. 问题 ...

  2. leetcode第一刷_Permutations II

    当有反复元素的时候呢? 不用拍脑袋都会想到一种方法,也是全部有反复元素时的通用处理方法,维护一个set,假设这个元素没增加过就增加,增加过了的忽略掉.可是,在这道题上这个通用方法竟然超时了! 怎么办? ...

  3. leetcode第一刷_Populating Next Right Pointers in Each Node II

    很自然的推广,假设去掉全然二叉树的条件呢?由于这个条件不是关键,因此不会影响整体的思路.做法依旧是每次找到一层的起点,然后一层一层的走. 假设是全然二叉树的话,每层的起点就是上一层起点的左孩子,兄弟之 ...

  4. leetcode第一刷_Merge Intervals

    看到这个题我就伤心啊,去微软面试的时候,第一个面试官让我做的题目就是实现集合的交操作,这个集合中的元素就像这里的interval一样.是一段一段的.当时写的那叫一个慘不忍睹.最后果然被拒掉了. .好好 ...

  5. leetcode第一刷_Gray Code

    说到格雷码,应该没人不知道,详细它有什么用,我还真不是非常清楚,我室友应该是专家.生成的规律不是非常明显,之前看到帖子讲的,这会儿找找不到了.. 思想是这种,假设有n位,在第2^(n-1)个编码以下画 ...

  6. leetcode第一刷_Combination Sum Combination Sum II

    啊啊啊啊.好怀念这样的用递归保存路径然后打印出来的题目啊.好久没遇到了. 分了两种,一种是能够反复使用数组中数字的,一种是每一个数字仅仅能用一次的.事实上没有多大差别,第一种每次进入递归的时候都要从头 ...

  7. leetcode第一刷_Count and Say

    水题. 描写叙述的还挺麻烦的,实际上就是纸老虎,用两个string,一个存上一轮的结果,一个用来更新出这一轮的结果,每次扫描上一轮,统计一个字符出现的次数,然后把这个次数和字符增加到这一轮的字符串中就 ...

  8. leetcode第一刷_Jump Game

    这个题事实上非常easy的,我一開始想复杂了,它没要求记录路径,事实上仅仅要看一下每一步之后所能延伸到的最远的位置就能够了,在这一个最远位置前面的那些位置,都是能够到达的,假设扫到了某个i,它大于当前 ...

  9. leetcode第一刷_Triangle

    非常easy的一道DP,看到空间限制是O(N)的,不要习惯性的以为是要保存每一行的最小值,不难想到是要保存一行其中各个数为路径终点时的和的大小.当算到最后一行时,就是从顶部究竟部以这个底部位置为终点的 ...

随机推荐

  1. C#反射调用外部Dll,执行其中异步函数并取返回值

    using System.Reflection; 1.载入Dll Assembly asm=Assembly.LoadFile(FullPath);//FullPath 为Dll所在位置的全路径. 2 ...

  2. iOS系统库头文件中NS_AVAILABLE相关

    转载: NS_AVAILABLE_IOS(5_0) 这个方法可以在iOS5.0及以后的版本中使用,如果在比5.0更老的版本中调用这个方法,就会引起崩溃. NS_DEPRECATED_IOS(2_0, ...

  3. Android - fragment之间数据传递

    <Fragment跳转时传递参数及结果回传的方法> <Fragment详解之五——Fragment间参数传递> <Android解惑 - 为什么要用Fragment.se ...

  4. Java springmvc 统一异常处理的方案

    前言:为什么要统一异常处理?经常在项目中需要统一处理异常,将异常封装转给前端.也有时需要在项目中统一处理异常后,记录异常日志,做一下统一处理. Springmvc 异常统一处理的方式有三种. 一.使用 ...

  5. C#的字节与流

    计算机中文件有很多种,我们知道实际存在计算机中的都是二进制.这里我记录了通过流对文件的读取操作. 一.首先在这里简单涉及下位,字节,字符的概念. 位(bit):可以表示0或1: 字节(byte):由8 ...

  6. UVAlive6800The Mountain of Gold?(负环)

    题意 题目链接 问从\(0\)出发能否回到\(0\)且边权为负 Sol 先用某B姓算法找到负环,再判一下负环上的点能否到\(0\) #include<bits/stdc++.h> #def ...

  7. 【代码笔记】iOS-cell自动变化大小

    一,效果图. 二,工程图. 三,代码. RootViewController.h #import <UIKit/UIKit.h> @interface RootViewController ...

  8. 详解php 获取文件名basename()函数的用法

    PHP 中basename()函数给出一个包含有指向一个文件的全路径的字符串,此函数返回基本的文件名,本篇文章收集了关于使用PHP basename()函数获取文件名的几篇文章,希望对大家理解使用PH ...

  9. Eclipse创建第一个Servlet(Dynamic Web Project方式)、第一个Web Fragment Project(web容器向jar中寻找class文件)

    创建第一个Servlet(Dynamic Web Project方式) 注意:无论是以注解的方式还是xml的方式配置一个servlet,servlet的url-pattern一定要以一个"/ ...

  10. python学习:数据类型检查

    函数调用时可能会出现数据类型不匹配的问题,为了保证代码的鲁棒性,最好加上数据类型检查. 应用举例: if not isinstance(x, (int, float)):      raise Typ ...