问题描述
在n*n的棋盘上放n个皇后,每两两不同行,不同列,不同对角线。求合法方案数
思路
直接列举是不太现实的,复杂度非常的大,解决方法是采用行组合的方式。
从左到右,每一个数字是第n列的第i行占有格子,则可表示为i,i,i…,i这样总共只有n!个排列。
我们只需要根据每一个排列,内部判断是否有处于同一条对角线的情况即可
具体见《笔记》P116
解法
- 朴素算法(暴力法):不考虑任何优化直接验证的方法
int count = 0;void generateP(int index){for(int x = 1; x <= n; x++){if(hashTable[x]==false){//如果该数字还没放进排列中P[index] = x;//就令当前位为该数字hashTable[x] = true;//标志该数字已经放入排列generateP(index + 1);//解决之后的数字hashTable[x] = false;//返回后因为要进行下一轮排列,所以置回false}}//排列生成完毕if(index == n + 1){//递归边界,排列结束bool flag = true;//默认合法标志for(int i = 1; i <= n; i++){//i, j代表任意两个皇后for(int j = i + 1; j <= n; j++){if(abs(i-j)==abs(P[i] - P[j])){//这行是判断是否处于同一对角线,需要记忆一下flag = false;//不符合,置false}}}if(flag) count++;return;}}
- 回溯法:指由于一些事实导致不需要任何一个字问题递归就可以直接返回上层的情况
void generateP(int index){if(index == n + 1){//边界返回count++;return;}for(int x = 1; x <= n; x++){//第x行if(hashTable[x] == false){//第x行还没有皇后bool flag == true;for(int pre = 1; pre < index; pre++){//遍历之前的皇后if(abs(index - pre) == abs(x - P[pre])){flag = false;break;}}if(flag){//如果不需要回溯那么放一个皇后P[index] = x;hashTable[x] = true;generate(index + 1);hashTable[x] = false;}}}}
