N皇后问题 回溯非递归算法 C++实现2
运行结果
代码如下
#include <bits/stdc++.h>
using namespace std;
const int MAX = ;
const char *LINE32 = "--------------------------------";
const bool PRINT_DETAILS = false;
long long n, cnt = ;// n表示皇后数量,cnt表示方案数量
int vis[][*MAX+];
//v[0][]、v[1][]、v[2][]分别表示列、主对角线、副对角线是否存在皇后
// 为0时表示无皇后,非0则表示有,且v[0][4]=2表示第四列第二行存在皇后 void print() {
cout << LINE32 << endl;
cout << "第" << cnt << "个方案: " << endl;
for (int i = ; i <= n; i++) {
if (i != ) {
cout << ", ";
}
cout << "(" << vis[][i] << "," << i << ")";
}
cout << endl;
for (int i = ; i <= n; i++) {
for (int j = ; j <= n; j++) {
if (vis[][j] != i) {
cout << 'x';
} else {
cout << 'Q';
}
}
cout << endl;
}
} bool check(int row, int col) {// 检查是否可以在(row,col)这个坐标放置皇后
return !(vis[][col] || vis[][row-col+n] || vis[][row+col]);
}
void place(int x, int y, int *r2c) {// 在(x,y)的位置上放置皇后
vis[][y] = x;
vis[][x-y+n] = vis[][x+y] = ;
r2c[x] = y;
}
void remove(int x, int y) {// 移除(x,y)位置上的皇后
vis[][y] = vis[][x-y+n] = vis[][x+y] = ;
}
void solve(int n) {// 与非递归实现1的不同点在于,A.使用了vis[3][]加快了判断,B.回溯的具体操作是在vis[3][]上而不是r2c[]上
int r2c[n+];// 存放各行所放位置的列数,其实类似于递归实现1和非递归实现1中使用的queen[]
r2c[] = ;// 这里要初始化,否则最后要退出下面的循环时会数组越界,受影响的代码是63_42
int row = , col = ;
place(row, col, r2c);
row = ;//在(1,1)的位置上放置一个皇后,然后进入循环,开始寻找第二行放置的位置
while (row > ) {// row的值最后为0,因为所有情况都检查完时,第一行往上回溯,row值就为0
if (row > n) {
// 找到一个解,之后需要向上回溯:移除上一行的皇后,从上一行的下一列尝试放置皇后
cnt++;
if (PRINT_DETAILS) {
print();
}
row--;// row返回上一行
remove(row, r2c[row]);// 移除上一行中的皇后
col = r2c[row]+;// 此时的(row,col)为下一次尝试放置的位置
} else if (col > n) {
// 当前row行中的每一个位置都尝试并放置了,回溯
row--;
remove(row, r2c[row]);
col = r2c[row]+;
} else if (check(row, col)) {
// 找到一个符合的位置
place(row, col, r2c);// 放置皇后
row++;// 查找下一行放置的位置
col = ;// 并且是从第一列开始放置
} else {
// 这一列不符合,查找下一列
col++;
}
}
} int main() {
// input
cout << "输入皇后个数: ";
cin >> n;
// compute
clock_t begin = clock();
solve(n);
clock_t end = clock();
// output
cout << LINE32 << endl;
cout << n << "皇后问题一共有" << cnt << "种解决方案" << endl;
cout << LINE32 << endl;
cout << "回溯非递归算法实现2 解决" << n << "皇后问题耗时" << /*end-begin << "打点" <<*/(double)(end-begin)/CLOCKS_PER_SEC << "s" << endl;
return ;
}
// 14 3~5s
与回溯递归算法实现2对比
回溯递归算法实现2运行情况
回溯非递归算法实现2运行情况
非递归实现还是比递归实现慢一点,有点不符合预期。
N皇后问题 回溯非递归算法 C++实现2的更多相关文章
- 理解 Hanoi 汉诺塔非递归算法
汉诺塔介绍: 汉诺塔(港台:河内塔)是根据一个传说形成的数学问题: 最早发明这个问题的人是法国数学家爱德华·卢卡斯. 传说越南河内某间寺院有三根银棒,上串 64 个金盘.寺院里的僧侣依照一个古老的预言 ...
- N个数全排列的非递归算法
//N个数全排列的非递归算法 #include"stdio.h" void swap(int &a, int &b) { int temp; temp = a; a ...
- 八皇后问题-回溯法(MATLAB)
原创文章,转载请注明:八皇后问题-回溯法(MATLAB) By Lucio.Yang 1.问题描述 八皇后问题是十九世纪著名数学家高斯于1850年提出的.问题是:在8*8的棋盘上摆放8个皇后,使其不能 ...
- C# 递归与非递归算法与数学公式
1.递归 递归:程序调用自身的编程技巧称为递归(recursion). 优点是:代码简洁,易于理解. 缺点是:运行效率较低. 递归思想:把问题分解成规模更小,但和原问题有着相同解法的问题. 1)下面是 ...
- 汉诺塔问题(The Tower of Hanoi)的递归算法与非递归算法
非递归算法: 根据圆盘的数量确定柱子的排放顺序: 若n为偶数,按顺时针方向依次摆放 A B C: 若n为奇数,按顺时针方向依次摆放 A C B. 然后进行如下操作: (1)按顺时针方向把圆盘1从现在的 ...
- 二叉树(9)----打印二叉树中第K层的第M个节点,非递归算法
1.二叉树定义: typedef struct BTreeNodeElement_t_ { void *data; } BTreeNodeElement_t; typedef struct BTree ...
- 【转】Java实现折半查找(二分查找)的递归和非递归算法
原创作品,允许转载,转载时请务必以超链接形式标明文章 原始出处 .作者信息和本声明.否则将追究法律责任.http://wintys.blog.51cto.com/425414/94051 Java二分 ...
- 八皇后,回溯与递归(Python实现)
八皇后问题是十九世纪著名的数学家高斯1850年提出 .以下为python语句的八皇后代码,摘自<Python基础教程>,代码相对于其他语言,来得短小且一次性可以打印出92种结果.同时可以扩 ...
- N皇后问题--回溯法
1.引子 中国有一句古话,叫做“不撞南墙不回头",生动的说明了一个人的固执,有点贬义,但是在软件编程中,这种思路确是一种解决问题最简单的算法,它通过一种类似于蛮干的思路,一步一步地往前走,每 ...
随机推荐
- MacOS下的渗透测试工具
信息收集工具 工具名称 安装命令 CeWL brew install sidaf/pentest/cewl dirb brew install sidaf/pentest/dirb dnsrecon ...
- python 清空list的几种方法
本文介绍清空list的四种方法,以及 list=[ ] 和 list.clear() 在使用中的区别(坑). 1.使用clear()方法 lists = [1, 2, 1, 1, 5] lists.c ...
- [尊老爱幼] Queen
You are given a rooted tree with vertices numerated from 1 to n . A tree is a connected graph withou ...
- Asp.Net Core 学习教程2、使用ASP.NET Core中的RazorPages
1.创建一个Asp.Net Core Web应用程序 1.1.打开VS2019 新建项目 1.2.选好项目位置后进入线面界面,选择Web应用程序 1.3.进去的页面结构如下 Pages 文件夹:包含 ...
- python plt 色卡
https://blog.csdn.net/Strive_For_Future/article/details/100151261 plt 绘图时通常需要各种颜色,还需要去介绍文档找,很麻烦,这里把p ...
- python 产生随机数,随机字符串
import randomimport string#随机整数:print random.randint(1,50)#随机选取0到100间的偶数:print random.randrange(0, 1 ...
- 累加数的贡献 CodeForces - 1213D2
题意: 第一行输入n,k,表示有n个数,可以进行整除2操作,要是数组有k个相等的数,最少需要几次操作. 思路: 用一个数组记录每一个数出现的次数,如果一开始大于等于k,直接输出0,否则对这n个数进行从 ...
- FME中矢量裁剪
- Java中如何调用静态方法
Java中如何调用静态方法: 1.如果想要调用的静态方法在本类中,可直接使用方法名调用 2.调用其他类的静态方法,可使用类名.方法名调用 关于静态方法能被什么调用 1.实例方法 2.静态发放
- Tail Call
一.什么是尾调用 尾调用(Tail Call)是函数式编程的一个重要概念. 一个函数里的最后一个动作是返回一个函数的调用结果,用简单的一句话描述就是"在函数的最后一步调用函数". ...