1756:八皇后

题目

总时间限制: 1000ms 内存限制: 65536kB

描述

会下国际象棋的人都很清楚:皇后可以在横、竖、斜线上不限步数地吃掉其他棋子。如何将8个皇后放在棋盘上(有8 * 8个方格),使它们谁也不能被吃掉!这就是著名的八皇后问题。
对于某个满足要求的8皇后的摆放方法,定义一个皇后串a与之对应,即a=b1b2...b8,其中bi为相应摆法中第i行皇后所处的列数。已经知道8皇后问题一共有92组解(即92个不同的皇后串)。
给出一个数b,要求输出第b个串。串的比较是这样的:皇后串x置于皇后串y之前,当且仅当将x视为整数时比y小。

输入

第1行是测试数据的组数n,后面跟着n行输入。每组测试数据占1行,包括一个正整数b(1 <= b <= 92)

输出

输出有n行,每行输出对应一个输入。输出应是一个正整数,是对应于b的皇后串。

样例输入

2
1
92

样例输出

15863724
84136275

题意

输入数据组数和b,输出八皇后的第b个串

思路

这题可以用深度优先搜索和递归,用数组a来表示棋盘,其中a[i][j]=1表示在第i行第j列放置了皇后。
用数组sum 来存储所有找到的解,其中 sum[k][i] 表示第k个解中第i行皇后所在的列号。
从第1行开始,逐行尝试放置皇后,对于每一行,尝试所有8个可能的列,位置对于每个位置,检查是否安全(不会被其他皇后攻击)
如果当前行找不到安全位置,则回溯到上一行,撤销上一行的皇后放置,尝试下一个可能的位置,接着继续向下搜索。
j函数检查三个方向:第y列,(x,y)位置的左上方向对角线,(x,y)位置的右上方向检查对角线。
当8个皇后全部放置,计数器ans加1,用sum数组记录每个皇后所在的列号。

代码

#include<bits/stdc++.h>
using namespace std;
int n,a[100][100]={0},b,sum[100][100]={0},ans=0;
bool j(int x,int y){//检查函数
   //检查列
	for(int i=1;i<=8;i++){
		if(a[i][y]==1) return false;
	}
	//检查左上对角线
	for(int i=x,j=y;i>=1&&j>=1;i--,j--){
		if(a[i][j]==1) return false;
    }
	//检查右上对角线
    for(int i=x,j=y;i>=1&&j<=8;i--,j++){
		if(a[i][j]==1) return false;
	}
	return true;//如果所有检查都通过,说明位置安全
}
//搜索函数
void d(int x){
    //如果已经成功放置了8个皇后就终止
	if(x>8){
		ans++;//解的数量加1
		for(int i=1;i<=8;i++){
			for(int j=1;j<=8;j++){
				if(a[i][j]==1){
					sum[ans][i]=j;//记录第ans个解中第i行皇后在第j列
					break;
				}
			}
		}
		return;//返回上一层递归
	}
	 //尝试在当前行的每一列放置皇后
	for(int y=1;y<=8;y++){
		if(j(x,y)==true){//如果当前位置安全
			a[x][y]=1;//放置皇后
			d(x+1);//递归放置下一行的皇后
			a[x][y]=0;//撤销当前放置的皇后
		}
	}
}
int main(){
	cin>>n;
	d(1);//从第1行开始生成
	for(int i=1;i<=n;i++){
		cin>>b;
		//输出第b个解
		for(int j=1;j<=8;j++){
			cout<<sum[b][j];
		}
		cout<<endl;
	}
	return 0;
}
posted @ 2025-09-19 21:15  骁翊  阅读(14)  评论(0)    收藏  举报