让人纠结的N皇后问题

       因为自己的手指刚做了一个激光手术,故而没有办法去享受美妙的五一假期,于是宅在家中练习一下回溯算法,选的题目是常见的N皇后问题,原以为很简单就可以解决的事情,没想到衍生除了较多的问题,现在一一记录下来,当然还有没有解决的问题值得以后自己思考。

       首先自己参照别人的思想写了一段最为原始的代码,如下:

#include <iostream.h>
#include <math.h>

#define MAX 4
int Queen[MAX];
int sum = 0;
//打印
void printOut() {
	for(int m=0;m<MAX;m++)
		cout<<Queen[m];
    cout<<endl;	

}

//判断第n个皇后放上去是否合法
int IsValid(int n) {
	for(int i=0;i<n;i++) {
		if(Queen[n] == Queen[i] || abs(Queen[n]-Queen[i]) == n-i)
			return 0;
	}
	return 1;
}
//在第i行放置皇后
void PutQueen(int i) {
		for(int j=0;j<MAX;j++) {
			if(i == MAX) {
				sum++;
				cout<<sum<<endl;
				printOut();
			}
			Queen[i] = j;
			if(IsValid(i)) 
				PutQueen(i+1); //递归
	}
}

int main() {

	PutQueen(0);
	cout<<"The count: "<<sum<<endl;
	return 0;
}

但是算法的运行相当不尽如人意,至少在统计总个数的变量上就表现得太差了

1
1302
1
1302
2
1302
3
1302
4
2031
1
2031
2
2031
3
2031
The count: 3
Press any key to continue

通过观察可以发现有两个问题,问题一:皇后的位置很强的重复性,也就是以MAX为周期输出,输出的皇后位置可以知道有可能出现了越界的问题(奇怪的是越界之后的数组都初始化Queen[MAX] = j)问题二,sum的变化相当奇怪,经过调试可以发现刚开始调用PutQueen(4)时,即当i = MAX时,输出sum = 1,以及Queen[] = {1302};但是,接下来奇怪的事情出现了,当跳出if(i == MAX)语句时(这时候其实已经出现了越界的情况),sum被置为0,相当的奇怪。接下来一系列的执行,等到再一次调用PutQueen(4)时,第一次输出是基于前一次的sum = 3, 执行自增后,输出sum = 4,当跳出if(i == MAX)语句时(这时候其实已经出现了越界的情况),sum被置为0,相当的摸不着头脑。这可是全局变量啊。

$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$

注意:插播一段对于sum出现变化“11234123”的原因,今天坐公交思索这个问题,sum作为一个全局变量没有出现sum++的自增而出现类周期这样的变化肯定是由于赋值导致的,到底什么地方出现了赋值,唯一可能的地方就是越界,于是晚上回到宿舍就直接进入内存的调试阶段,通过观察内存的变化验证自己的猜想。

验证过程:设置断点

让人纠结的N皇后问题_第1张图片

确定相关变量的内存

       经过反汇编得到:Queen[MAX]的内存起始地址:0x012D9138;sum的地址为:0x012D9148

调试关键语句,追踪sum变化
调用PutQueen(4)时,执行完sum++,这时候sum = 1,内存0x012D9148为01

让人纠结的N皇后问题_第2张图片

执行Queen[4] = 0语句,Queen[4]其实已经越界,它的赋值直接覆盖了sum的内存,也就是说sum被赋值为0,事实也是如此,看内存0x012D9148为00让人纠结的N皇后问题_第3张图片

此时sum=0; 经过for循环 j=1;但是经过sum++将sum = 1,输出。即

让人纠结的N皇后问题_第4张图片

Queen[4] = 1,再次将sum = 1(不过这时候显示的跟上面是一样的,但是sum确实经过了赋值为1这个过程,不能忽视),经过for循环j=2,执行sum++将sum = 2,输出。即让人纠结的N皇后问题_第5张图片

Queen[4] = 2,再次将sum = 2(不过这时候显示的跟上面是一样的,但是sum确实经过了赋值为2这个过程,不能忽视),经过for循环j=3,执行sum++将sum = 3,输出。

让人纠结的N皇后问题_第6张图片

如此下去即可得到整个sum的变化为11234123,验证正确,sum的变化很大程度是由于Queen[4] = j越界赋值造成的。

总结一下:编程一定要注意数组越界问题,肯定会赋值影响,这就是一个例子,同时回溯问题一定要注意控制规模,下面也会提起。这就引起了一个新的问题,出现上述问题是因为编译器在给变量分配内存的时候,sum紧紧的挨着Queen[]。于是我在声明的时候更改了两者的顺序,但是结果仍然是sum在Queen[]数组后面,我的问题是C编译器怎么根据变量的声明分配内存的呢?

插播结束。

$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$

首先解决越界的问题,修改函数PutQueen(int i);

void PutQueen(int i) {
		for(int j=0;j<MAX;j++) {
			if(i == MAX) {
				sum++;
				cout<<sum<<endl;
				printOut();
			}
			else {
				Queen[i] = j;
				if(IsValid(i)) 
					PutQueen(i+1); //递归
			}
		}
}

添加else语句,一定要这么做,否则很容易出现顺序执行的情况,即便不添加else也一定要在在if语句中添加返回或中断语句,此次得到的运行结果是:

1
1302
2
1302
3
1302
4
1302
5
2031
6
2031
7
2031
8
2031
The count: 8
Press any key to continue

我直接一头雾水了,这时候sum突然变得像全局变量的样子了,不会出现自己重置的现象,(我至今没有搞懂前面为什么会sum自己重置)。这时候发现结果中仍然是重复输出,这是因为在for循环中每次都会输出一次,于是我们在if语句中添加返回语句return;

void PutQueen(int i) {
		for(int j=0;j<MAX;j++) {
			if(i == MAX) {
				sum++;
				cout<<sum<<endl;
				printOut();
				return;
			}
			else {
				Queen[i] = j;
				if(IsValid(i)) 
					PutQueen(i+1); //递归
			}
		}
}

吁~这次得到结果:

1
1302
2
2031
The count: 2
Press any key to continue

¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$¥¥¥¥¥¥¥¥¥¥¥¥¥

自己的皇后问题就这样解决了,NO,这里面隐藏了太多的问题,归根结底说明自己的编程经验不足,同时也说明对于回溯算法理解还不够深入。

下面对于代码进行进一步的修改,对于PutQueen()函数的逻辑理解要有深入,,尽量在不出现越界的情形下就把结果输出(1.保证不越界 2.最后一行只要测试成功即输出)

更改后的代码如下:

void PutQueen(int i) {
 	if(i<MAX){
  		for(int j=0;j<MAX;j++) {
   			Queen[i] = j;
   			if(IsValid(i)) { 
    				if(i == MAX-1) {
     					sum++;
     					cout<<sum<<endl;
     					printOut();
    					return;
    				}
    				else
     					PutQueen(i+1); //递归
   			}
  		}
 	}
}


函数一开始就进行安全测试,保证不越界。同时保证只要对于每行的各列进行测试,只要合乎要求并且处在最后一行,就输出结果。经测试运行结果正确。

下面我们看一段更加符合回溯算法的代码: 

#include <iostream.h> #include <math.h>

#define M 4 int Queen[M];

int count = 0; void PutQueen(int k) {  if(k<M) {   for(int i=0;i<M;i++) {    Queen[k] = i;    bool IsValid = true;    for(int j=0;j<k;j++) {     if(Queen[k] == Queen[j] || abs(Queen[k]-Queen[j]) == k-j)      IsValid = false;    }    if(IsValid) {     if(k == M-1) {     count++;     cout<<"The count: "<<count<<endl;     for(int m=0;m<M;m++)      cout<<Queen[m];     cout<<endl;        }     else      PutQueen(k+1);    }   }  } } int main() {

 PutQueen(0);  cout<<"The count: "<<count<<endl;  return 0; }


这是根据ACM教程实现的回溯算法,可以看到对于规模的限制,以及关于if...else...语句的使用。(if语句后面要是有并行语句,一定要谨慎,否则就变成顺序执行了,这样很容易引起越界,同时关于回溯算法一定要有一个终止的语句) 

 

 

 

你可能感兴趣的:(让人纠结的N皇后问题)