POJ 2243 简单搜索 (DFS BFS A*)
题目大意:国际象棋给你一个起点和一个终点,按骑士的走法,从起点到终点的最少移动多少次。
求最少明显用bfs,下面给出三种搜索算法程序:
// BFS
#include<cstdio>
#include<queue>
#include<cstring>
using namespace std;
const int maxn=;
int r1,c1,r2,c2;
struct Node
{
int r,c;
Node(int r,int c):r(r),c(c){}
};
int vis[maxn][maxn];
int dist[maxn][maxn];
queue<Node> Q;
int dr[]={-,-,-,-,,,,};
int dc[]={-,,,-,,-,,-};
int BFS()
{
if(r1==r2&&c1==c2) return ;
while(!Q.empty()) Q.pop();
memset(vis,,sizeof(vis));
vis[r1][c1]=;
dist[r1][c1]=;
Q.push(Node(r1,c1));
while(!Q.empty())
{
Node node=Q.front();Q.pop();
int r=node.r,c=node.c;
for(int d=;d<;d++)
{
int nr=r+dr[d];
int nc=c+dc[d];
if(nr>=&&nr< && nc>= &&nc< &&vis[nr][nc]==)
{
if(nr==r2&&nc==c2) return dist[r][c]+;
dist[nr][nc]=+dist[r][c];
Q.push(Node(nr,nc));
vis[nr][nc]=;
}
}
}
return -;
}
int main()
{
char str1[],str2[];
while(scanf("%s%s",str1,str2)==)
{
r1=str1[]-'';
c1=str1[]-'a';
r2=str2[]-'';
c2=str2[]-'a';
printf("To get from %s to %s takes %d knight moves.\n",str1,str2,BFS());
}
return ;
}
DFS:
注意visited结点,如果步数较小也继续搜索
// DFS
#include<cstdio>
#include<algorithm>
#include<cstring>
using namespace std;
const int maxn=;
int r1,c1,r2,c2;
struct Node
{
int r,c;
Node(int r,int c):r(r),c(c){}
};
int vis[maxn][maxn];
int dist[maxn][maxn];
int dr[]={-,-,-,-,,,,};
int dc[]={-,,,-,,-,,-};
int ans;//最终答案
void DFS(int r,int c,int len)
{
if(ans<=len || r< || r>= || c< || c>=) return ;//剪枝
if(r==r2&&c==c2) ans=min(ans,len);
if(vis[r][c]== || dist[r][c]>len)
{
vis[r][c]=;
dist[r][c]=len;
}
else if(vis[r][c]== && len >= dist[r][c] )
return ;
for(int d=;d<;d++)
{
int nr=r+dr[d];
int nc=c+dc[d];
DFS(nr,nc,len+);
}
}
int main()
{
char str1[],str2[];
while(scanf("%s%s",str1,str2)==)
{
r1=str1[]-'';
c1=str1[]-'a';
r2=str2[]-'';
c2=str2[]-'a';
memset(vis,,sizeof(vis));
ans=;
DFS(r1,c1,);
printf("To get from %s to %s takes %d knight moves.\n",str1,str2,ans);
}
return ;
}
A*算法:
g函数为沿路径从起点到当前点的移动耗费(经过的步数),启发函数h为当前格子到终点横坐标差与纵坐标差的和(曼哈顿距离),用优先队列保存每个状态按照g+h排序。通常,我们令水平或者垂直移动的耗费为10,对角线方向耗费为14。取这些值是因为沿对角线的距离是沿水平或垂直移动耗费的的根号2,或者约1.414倍。为了简化,可以用10和14近似。
bfs
#include <cstring>
#include <iostream>
#include <cmath>
#include <algorithm>
#include <vector>
#include <queue>
#include <cstdio> using namespace std; const int maxn = ; struct Node{
int x,y,step;
int f,g,h;
// g函数为走到当前状态的经过的步数,启发函数h为当前格子到终点横坐标差与纵坐标差的和,用优先队列保存每个状态按照g+h排序
bool operator < (const Node & tmp) const{
return f > tmp.f;
}
}node; bool visd[maxn][maxn];
priority_queue<Node> Que;
int dirs[][]={{-,-},{-,},{,-},{,},{-,-},{-,},{,-},{,}};
int start_x,start_y,end_x,end_y,ans; bool isIN(int x,int y)
{
if(x>=||x<)return false;
if(y>=||y<)return false;
return true;
}
void init()
{
memset(visd,false,sizeof(visd));
while(!Que.empty())Que.pop();
Node S;
S.x=start_x;S.y=start_y;
S.step=;S.g=;S.h=(abs(end_x-start_x)+abs(end_y-start_y))*;
S.f=S.g+S.h;
visd[S.x][S.y]=true;
Que.push(S);
ans=-;
} void Astar()
{
Node A,B;
while(!Que.empty())
{
A=Que.top();Que.pop();
if(A.x==end_x&&A.y==end_y)
{
ans=A.step;
break;
}
for(int i=;i<;i++)
{
int xx=dirs[i][]+A.x;
int yy=dirs[i][]+A.y;
if(isIN(xx,yy)==false||visd[xx][yy])continue;
B.x=xx;B.y=yy;
B.step=A.step+;
B.g=A.g+;
B.h=(abs(end_y-yy)+abs(end_x-xx))*;
B.f=B.g+B.h;
visd[B.x][B.y]=true;
Que.push(B);
}
}
} int main()
{
char line[];
while(gets(line))
{
start_x=line[]-'a';start_y=line[]-'';
end_x=line[]-'a';end_y=line[]-'';
init();
Astar();
printf("To get from %c%c to %c%c takes %d knight moves.\n",line[],line[],line[],line[],ans);
} return ;
}
POJ 2243 简单搜索 (DFS BFS A*)的更多相关文章
- 简单搜索dfs, 简单的修剪搜索
选择最合适的语言做一个项目是非常重要的.但,熟练的掌握自己的武器,这也是非常重要的. ========================================================= ...
- poj 1426 Find The Multiple (简单搜索dfs)
题目: Given a positive integer n, write a program to find out a nonzero multiple m of n whose decimal ...
- POJ - 1321 棋盘问题 简单搜索 dfs 格子
点这里去看题 思路:本题的难点在k<n的情况,所以我们可以另dfs中的两个参数分别代表起始行和待放棋子个数(待放棋子只能放在起始行后面的行),然后用一个c[8]来表示每一列放旗子的情况来判断列不 ...
- kb-01-a<简单搜索--dfs八皇后问题变种>
题目描述: 在一个给定形状的棋盘(形状可能是不规则的)上面摆放棋子,棋子没有区别.要求摆放时任意的两个棋子不能放在棋盘中的同一行或者同一列,请编程求解对于给定形状和大小的棋盘,摆放k个棋子的所有可行的 ...
- poj2386(简单的dfs/bfs)
题目链接:http://poj.org/problem?id=2386 Description Due to recent rains, water has pooled in various pla ...
- poj 1167 简单搜索
这题主要是注意好限定的条件 条件1:每个公交车都至少要到达两次 条件2:公交车相同时间和相同间隔是属于两种车辆 条件3:不同的车可能到达时间相同 上述都是深搜的重要条件: #include<al ...
- 棋盘问题 简单搜索DFS
Description 在一个给定形状的棋盘(形状可能是不规则的)上面摆放棋子,棋子没有区别.要求摆放时任意的两个棋子不能放在棋盘中的同一行或者同一列,请编程求解对于给定形状和大小的棋盘,摆放k个棋子 ...
- 连连看(简单搜索)bfs
连连看Time Limit: 20000/10000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others)Total Submis ...
- 搜索进阶课件,视频,代码(状态压缩搜索,折半搜索,dfs,bfs总结)
链接:https://pan.baidu.com/s/1-svffrprCOO4CtQoCTQ9hQ 提取码:h909 复制这段内容后打开百度网盘手机App,操作更方便哦
随机推荐
- python3+selenium入门03-操作谷歌浏览器
操作谷歌浏览器同样也需要下载相应的driver插件,官网下载.也可以从我的网盘下载,不过可能不是最新的,网盘地址.打开谷歌浏览器的操作和火狐的差不过. from selenium import web ...
- git与eclipse集成之保存快照
1.1. 保存快照 在个分支进行编码,然后需要紧急切换到另外一个分支进行快速修复一个问题,此时可以先将当前分支的修改进行保存快照. 在分支A进行编码,保存快照 切换到另外分支B进行修改 切换回A分支继 ...
- NOIP模拟赛10 题解
t3: 题意 给你一棵树,然后每次两种操作:1.给一个节点染色 : 2. 查询一个节点与任意已染色节点 lca 的权值的最大值 分析 考虑一个节点被染色后的影响:令它的所有祖先节点(包括自身)的所有除 ...
- LA 4108 (线段树)
区间更新 + 统计更新长度 稍稍不注意就T了 #include<bits/stdc++.h> #define lson l, m, rt<<1 #define rson m+1 ...
- Laravel 5.2数据库--多个关联关系,带条件约束的渴求式加载的问题
### 今天在连表获取数据的时候,老是获取不到想要的,确实有点无力适从的感觉. 归根到底,还是对laravel不够熟悉,至少是数据库操作那块. ### 问题是这样的: 我想要通过连表中间表,拿中间表的 ...
- 如何将本地项目上传到Github
看了这篇文章觉得写的很详细很适合初学者 提供给大家参考下. http://blog.csdn.net/zamamiro/article/details/70172900 注意如果第二次git pus ...
- Linux IO实时监控iostat命令
简介 iostat主要用于监控系统设备的IO负载情况,iostat首次运行时显示自系统启动开始的各项统计信息,之后运行iostat将显示自上次运行该命令以后的统计信息.用户可以通过指定统计的次数和时间 ...
- Golang服务器热重启、热升级、热更新(safe and graceful hot-restart/reload http server)详解
服务端代码经常需要升级,对于线上系统的升级常用的做法是,通过前端的负载均衡(如nginx)来保证升级时至少有一个服务可用,依次(灰度)升级. 而另一种更方便的方法是在应用上做热重启,直接更新源码.配置 ...
- (转)整理 node-sass 安装失败的原因及解决办法
转载地址:https://segmentfault.com/a/1190000010984731
- swift 学习- 21 -- 类型转换
// 类型转换 可以判断实例的类型, 也可以将实例看做其父类的或者子类的实例 // 类型转换在 Swift 中使用 is 和 as 操作符实现, 这两个操作符提供了一种简单达意的方式去检查值的类型 或 ...