今天给大家介绍一下八皇后问题。网上介绍这个的已经很多很多了,我今天主要想说的是一种比较酸爽的解题方法。用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