回溯算法————n皇后、素数串
回溯就是算法是搜索算法中一种控制策略,是一个逐个试探的过程。在试探的过程中,如果遇到错误的选择,就会回到上一步继续选择下一种走法,一步一步的进行直到找到解或者证明无解为止。
如下是一个经典回溯问题n皇后的解答树:
下面就从n皇后说起:
【问题描述】
在n×n的国际象棋盘上,放置n个皇后,使任何一个皇后都不能吃掉另一个,需满足的条件是:同一行、同一列、同一对角线上只能有一个皇后。求所有满足要求的放置方案。4皇后的放置方案如下:
【输入】一个正整数n。
【输出】每行代表一种放置方案:第i行的第一个数是i,表示第i种方案,后面一个冒号,然后是用空格隔开的n个数,其中第i个数x[i]表示第i行上的皇后放在第x[i]列;最后一行:一个整数,表示方案总数。
【样例输入】
4
【样例输出】
1:2 4 1 3
2:3 1 4 2
2
这个题目简直是太经典了,一提到回溯第一个想到的就是它。这个题非常的通俗易懂,就是在一个n*n的棋盘上满足条件地放置n个皇后,每种组合搜一遍就好了。在搜的过程中判断下一步的某种方向是否可以走,如果可以的话就进入下一层,如果不符合要求就搜下一个方向,搜完这一层的所有方向以后就回到上一层,继续搜上一层的下一个方向。
不知道大家看懂了没有,如果没看懂,看代码就懂了。。
上代码:
#include<iostream>
#include<cstdio> using namespace std; int n;
int sum=0;//解法存放个数
int a[100000];//存放皇后位置数据
bool b[1000000]={0},c[1000000]={0},d[10000000]={0};//b存储y方向,c、d存储对角线 void print()
{
sum++;
cout<<sum<<':';
for(int i=1;i<=n;i++)
{
cout<<a[i]<<' ';
}
cout<<endl;
} int search(int x)
{
for(int j=1;j<=n;j++)//遍历本层中每种方向
{
if((!b[j])&&(!c[x+j])&&(!d[x-j+n-1]))//如果满足条件
{
a[x]=j;//存入皇后数据
b[j]=1;//占领y轴
c[x+j]=1;//占领对角线
d[x-j+n-1]=1;//占领对角线
if(x==n)
{
print();//如果有完整解,输出
}
else
{
search(x+1);//如果没有继续找下一行
}
b[j]=0;//找完之后取消占领
c[x+j]=0;
d[x-j+n-1]=0;
}
}
} int main()
{
cin>>n;
search(1);//从第一个开始找
cout<<sum;
}
把思想带到代码里应该就看懂了吧。
回溯就是这样一层一层的找,找完本层返回上一层继续找,直到找到正确解为止。
继续再看一道题:
3、素数链
设计程序将1。。。n排成一排,使任意两个相邻的数的和为素数。第1个和最后一个的和也为素数.输出一种方案即可。
输入:
n (n<=100)
输出:
1到n的一个序列,中间用一个空格隔开。第一个数为1。
如果无解输出-1。
样例输入:
10
样例输出:
1 2 3 4 7 6 5 8 9 10
这道题和刚才那道n皇后的思路是一样的,遍历所有可能,在遍历的过程中判断,如果可以就继续搜下一层。
代码如下:
#include<cstdio>
#include<iostream> const int maxx=1000; using namespace std; int n;
int numguo[maxx]={0};
bool guo[maxx]={0},pguo=0;
int c=0; int sushu(int x)//验证素数的函数
{
//关于判断素数的题很早就做过,这里我就不解释了
int i=2;
while(i<x)
{
if(x%i==0)
return 0;//如果不是直接跳出返回0
i++;
}
return 1;//如果是的话返回1
} int panduan()
{
int t;
for(int i=1;i<n;i++)//判断是否达到要求
{
if(!sushu(numguo[i]+numguo[i+1]))//这里我直接放到函数里去验证
{
return 0;//如果不是素数就直接跳出返回0
}
}
if(!sushu(numguo[n]+numguo[1])) return 0;//注意别忘了最后一个和第一个数字的和是否为素数
return 1;//正确的话返回1
} int print()
{
c++;//可能方法+1
for(int i=1;i<=n;i++){
cout<<numguo[i]<<' ';
}
cout<<endl;
} int search(int x)
{
for(int i=1;i<=n;i++)//搜索本层中每种可能
{
if(!guo[i])
{
guo[i]=1;//占领
numguo[x]=i;//标记
if(x==n)//如果达到数量要求
{ if(panduan())//如果符合要求
{
if(!pguo)//如果没输出过
{
pguo=1;//输出过
print();//输出
return 0;//搜到就直接跳出
}
}
}
else search(x+1);
guo[i]=0;
numguo[x]=0;
}
}
} int main()
{
cin>>n;
search(1);//从第一个开始找
if(!c) cout<<"-1";//如果没可能就输出“-1”
return 0;
}
这些代码中我用了很多的函数,这样节约了代码长度,也更加方便(个人觉得)。
回溯就是这个中心思想,一步步往下搜,一层层的搜,直到搜完或找到结果为止。
其实上面两道题都是很水的基础,再往后还有回溯遍历图、树等等等等。。
ending。。。
回溯算法————n皇后、素数串的更多相关文章
- 回溯算法 - n 皇后问题
(1)问题描述 在 n × n 格的棋盘上放置彼此不受攻击的 n 个皇后.按照国际象棋的规则,皇后可以攻击与之处在同一行或同一列或同一斜线上的棋子.n 后问题等价于在 n × n 的棋盘上放置 n 个 ...
- 8皇后以及N皇后算法探究,回溯算法的JAVA实现,非递归,循环控制及其优化
上两篇博客 8皇后以及N皇后算法探究,回溯算法的JAVA实现,递归方案 8皇后以及N皇后算法探究,回溯算法的JAVA实现,非递归,数据结构“栈”实现 研究了递归方法实现回溯,解决N皇后问题,下面我们来 ...
- 8皇后以及N皇后算法探究,回溯算法的JAVA实现,递归方案
八皇后问题,是一个古老而著名的问题,是回溯算法的典型案例.该问题是国际西洋棋棋手马克斯·贝瑟尔于1848年提出:在8×8格的国际象棋上摆放八个皇后,使其不能互相攻击,即任意两个皇后都不能处于同一行.同 ...
- 回溯算法之n皇后问题
今天在看深度优先算法的时候,联想到DFS本质不就是一个递归回溯算法问题,只不过它是应用在图论上的.OK,写下这篇博文也是为了回顾一下回溯算法设计吧. 学习回溯算法问题,最为经典的问题我想应该就是八皇后 ...
- 回溯算法-C#语言解决八皇后问题的写法与优化
结合问题说方案,首先先说问题: 八皇后问题:在8X8格的国际象棋上摆放八个皇后,使其不能互相攻击,即任意两个皇后都不能处于同一行.同一列或同一斜线上,问有多少种摆法. 嗯,这个问题已经被使用各种语言解 ...
- C语言回溯算法解决N皇后问题
回溯算法的模型是 x++, not satisfy ? x-- : continue. 代码中x作列号,y[x]保存第x列上皇后放置的位置. #include<stdio.h> #incl ...
- 回溯算法 LEETCODE别人的小结 一八皇后问题
回溯算法实际上是一个类似枚举的搜索尝试过程,主要是在搜索尝试中寻找问题的解,当发现已不满足求解条件时,就回溯返回,尝试别的路径. 回溯法是一种选优搜索法,按选优条件向前搜索,以达到目的.但是当探索到某 ...
- 回溯算法——解决n皇后问题
所谓回溯(backtracking)是通过系统地搜索求解问题的方法.这种方法适用于类似于八皇后这样的问题:求得问题的一个解比较困难,但是检查一个棋局是否构成解很容易. 不多说,放上n皇后的回溯问题代码 ...
- JS算法之八皇后问题(回溯法)
八皇后这个经典的算法网上有很多种思路,我学习了之后自己实现了一下,现在大概说说我的思路给大家参考一下,也算记录一下,以免以后自己忘了要重新想一遍. 八皇后问题 八皇后问题,是一个古老而著名的问题,是回 ...
随机推荐
- Python调用C/C++动态链接库
Python调用C/C++动态链接库 2013年07月26日 ⁄ 综合 ⁄ 共 3219字 ⁄ 字号 小 中 大 ⁄ 评论关闭 吐槽(可略过):不知不觉,4月份毕业,5月份进入团队,已有7个月.大 ...
- spring的IOC,DI及案例详解
一:spring的基本特征 Spring是一个非常活跃的开源框架:它是一个基于Core来架构多层JavaEE系统的框架,它的主要目的是简化企业开发.Spring以一种非侵入式的方式来管理你的代码,Sp ...
- 【Android 错误记录】Conversion to Dalvik format failed with error 1 错误
错误原因:依赖的包中有冲突,比如依赖了同一个jar包的不同版本等 在以往测试的过程中,出现过几次这个问题,根本原因都是因为有冲突了,但是表现形式可能不一样 情况1: 有一个叫DemoAPP的工 ...
- Oracle表分区[转]
废话少说,直接讲分区语法. Oracle表分区分为四种:范围分区,散列分区,列表分区和复合分区. 一:范围分区 就是根据数据库表中某一字段的值的范围来划分分区,例如: create table gra ...
- [汇编语言]-第二章DEBUG
Debug查看CPU各种寄存器中得内容,内存的情况和在机器码级跟踪程序的运行. 1- 进入Debug xp 开始-运行 cmd 输入 debug 2- Debug功能 r 查看,改变CPU寄存器的内容 ...
- python列表推导和字典推导
代码如下: list = ['aaa','bbb','ccc','ddd'] dict = {key:value for value,key in enumerate(list)} print(dic ...
- JS 禁止浏览器右键菜单和刷新
<script language="javascript"> //禁止按键F5 document.onkeydown = function(e){ e = window ...
- 自学nodejs系列
好久没有搞nodejs了.nodejs也是更新的很快!!当初翻译的文档
- 使用Hashtable和List结合拼json数据
在做项目的时候,有时候需要向页面返回一个特定的json类型的数据,一般情况下会有下面的方法进行拼接: public String chongzhiList() throws Exception { L ...
- nginx安装及配置支持php的教程(全)
本文的实验环境为:Centos4.5,nginx版本为:nginx-0.7.26 pcre-7.8.tar.gz 正则表达式下载地址:ftp://ftp.csx.cam.ac.uk/pub/so ...