八皇后问题

今天给大家介绍一下八皇后问题。网上介绍这个的已经很多很多了,我今天主要想说的是一种比较酸爽的解题方法。用C++在10行内写出八皇后。解这种题还是得用回溯。
大家先看一下下面比较正常的解题代码吧。

#include <iostream>
#define N 8
using namespace std;
int arr[N+1]={0,0,0,0,0,0,0,0};
bool canPlace(int k){
    for(int i=1;i<k;++i)
        if(abs(arr[i]-arr[k])==k-i || arr[i]==arr[k])
            return false;
    return true;    
}
void print(){
    for(int i=1;i<N+1;i++)
        cout<<arr[i]<<" ";
    cout<<endl;
}
void eightQueen_1(){
    int index=1;
    while(index>=1){

        while(arr[index]<=7){
            arr[index]+=1;            // try next pos
            if(canPlace(index) && index==8)  //if can place this piece and quees counts to eight
                print();
            else if( canPlace(index) )      //if the piece can place at this pos
                index++;
        }
        arr[index]=0;
        --index;
    }
}
void eightQueen_2(int index){
    for(int i=1;i<9;++i){
        arr[index]=i;
        if(index==8 && canPlace(index))
            print();
        else
        if(canPlace(index))
            eightQueen_2(index+1);   //try to next pos 
    }
}
int main(int argc,char **argv){
//  eightQueen_1();
    eightQueen_2(1);
    return 0;
}

这个也比较简单,没什么好说的,大家注意看下面的。

#include <iostream>
#include <algorithm>
#include <bitset>
#include <numeric>
#include <utility>
int main() {
  for (int queens[] = {0,1,2,3,4,5,6,7}; ::std::next_permutation(queens,queens+8); )
    if ((::std::bitset<15>(::std::accumulate(queens,queens+8, ::std::make_pair(0, 0), [](::std::pair<int, int> a, int b){return ::std::make_pair((1<<(b+a.second))|a.first,a.second+1);}).first).count() == 8) && (::std::bitset<15>(::std::accumulate(queens, queens+8, ::std::make_pair(0, 0), [](::std::pair<int, int> a, int b){return ::std::make_pair((1<<(7+b-a.second))|a.first, a.second+1);}).first).count() == 8))
      ::std::cout << queens[0] << queens[1] << queens[2] << queens[3] << queens[4] << queens[5] << queens[6] << queens[7] << ::std::endl;
}

这个才是最酸爽的,哈哈!
思路呢,我简单地给大家提示一下,其实我也是看了好久才明白的。主要用了C++的lambda表达式和标准库提供的一些函数。next_permutation函数会产生 0 1 2 3 4 5 6 7的全排列。所以我们现在需要做的就是检测某个排列是否冲突。在同一行和同一列的情况已经不用考虑了,所以要检测呢就是检测是否在同一条对角线上。实际上用两个函数就可以表达是否在同一条对角线了咩。
y=x+b y=-x+b
提示到这了,大家自己想吧。

参考资料:http://www.zhihu.com/question/28543312
              http://blog.csdn.net/crayondeng/article/details/17174557

你可能感兴趣的:(算法)