八皇后-经典搜索

八皇后

http://www.nowcoder.com/questionTerminal/fbf428ecb0574236a2a0295e1fa854cb

不到30行代码解八皇后问题:
https://blog.csdn.net/csyifanZhang/article/details/105704431
↑更好的阅读体验

先来看一看洛谷的八皇后问题
在这里插入图片描述
有了这个图就显得比较清晰了,就是在8*8或者6*6的棋盘上放置8或者6个棋子,使得棋子所在行列个不相交。建议直接学数独游戏,基本上搜索难不过八数码和数独毕竟我做的题少,没遇到过 。这道题打眼一看就要来搜索,那么我们就要确定搜索的状态是什么?如何进行搜索?

如果我们使用dfs进行搜索,那么首先要保证的是条件:==每行每列每条对角线都没有连个皇后==,行和列还比较好处理,一旦给某个位置加上一个皇后,那么我们标记该行该列即可,比较麻烦的就是对角线的记录:

对角线的元素有如下的性质:

  • 对于一条从右上到左下的对角线,其上的棋子坐标应满足x+y为一定值;(比如1,66,1
  • 对于一条从左上到右下的对角线,其上的棋子坐标应满足x-y为一定值,为了避免负数的产生,代码中用x-y+n来储存数字,(比如1,16,6)
  • 而且上述两种方案在每条对角线上得到的值各不相同,因此可以用于标识对角线是否存在元素。

在这里插入图片描述
下面这一段搜索足够解决的n皇后问题,如果,则使用vector存储历史路径即可当然也有其他解决方案。

void dfs(ll x, string s) {//对第x行进行搜索
    if (x == n + 1) { res++; v.push_back(s); return; }//成功凑成n皇后
    for (int i = 1; i <= n; i++) {//尝试第一行的每个位置
        if (!row[x] && !col[i] && !eye1[x + n - i] && !eye2[x + i]) {
            row[x] = col[i] = eye1[x + n - i] = eye2[x + i] = 1; 
            s += i + '0';
            dfs(x + 1, s);//x行有了皇后 继续到下一行
            s.erase(s.end()-1);
            row[x] = col[i] = eye1[x + n - i] = eye2[x + i] = 0;//回溯
        }
    }
}

到这里这道题已经非常容易了,只需要指定n=8,然后输入b,直接输出v[b-1]即可。

//a:棋盘当前的状态  row[i]:第i行有元素  col[i]:第j列有元素
//eye1:主对角线(x+n-y定值)  eye2:次对角线(x+y定值)
ll row[MAX], col[MAX], eye1[MAX * 3], eye2[MAX * 3], n = 8, res = 0, b;
vector<string> v;

void dfs(ll x, string s) {//对第x行进行搜索
    if (x == n + 1) { res++; v.push_back(s); return; }//成功凑成n皇后
    for (int i = 1; i <= n; i++) {//尝试第一行的每个位置
        if (!row[x] && !col[i] && !eye1[x + n - i] && !eye2[x + i]) {
            row[x] = col[i] = eye1[x + n - i] = eye2[x + i] = 1; 
            s += i + '0';
            dfs(x + 1, s);//x行有了皇后 继续到下一行
            s.erase(s.end()-1);
            row[x] = col[i] = eye1[x + n - i] = eye2[x + i] = 0;//回溯
        }
    }
}

int main(){
    while (cin >> b) {
        res = 0; string s; v.clear();
        dfs(1, s);
        sort(v.begin(), v.end());
        cout << v[b - 1] << endl;
    }
}
全部评论

相关推荐

评论
1
收藏
分享

创作者周榜

更多
牛客网
牛客企业服务