思路:

先用bfs求出入口,宝物,出口,两两之间的最短距离。

在用dfs搜索所有情况,求出从入口走到出口能获得的最大价值。

我们要解决几个问题:1、求入口到第一个取宝物的地方的最短距离

2、求第i个取宝物的地方到第i+1个取宝物的地方的最短距离

3、求第n个取宝物的地方到出口的最短距离

4、保证以上3点能在时间L内实现的情况下,取得的宝石价值最大

熟悉两种搜索的优缺点:

BFS: 对于解决最短或最少问题特别有效,而且寻找深度小,但缺点是内存耗费量大(需要开大量的数组单元用来存储状态)。

DFS:对于解决遍历求和问题有效,对于问题搜索深度小的时候处理速度迅速,然而在深度很大的情况下效率不高

dfs剪枝:
1.step>time直接return。
2.ans==sum时就不用再搜了  因为已经到最大了。
3.如果搜到一个点,这个点以前已经搜过,而且现在到达这个点时珠宝价值比以前少而且走的步数却比以前多,就不用搜这个点了。
真是被自己傻到了 。。。花了一个小时写了一个代码如下,感觉自己真是太弱智了,居然放这么大的错误,忽略了两件宝物可以在一条路径上,所以 以后编程还是要考虑周全在下手 ,尤其是比赛的时候,一旦有大的漏洞,就会被 坑哭了。。。
错误代码:
#include"iostream"
#include"stdio.h"
#include"algorithm"
#include"cmath"
#include"string"
#include"string.h"
#include"queue"
#include"stack"
#include"vector"
#define mx 105
using namespace std;
int g[mx][mx];
int value[mx][mx];//存放每个点的价值
int time[mx][mx];//存放每个点的时间
int M[mx];//存放宝物的价值
int h,w,m,l,sx,sy,ex,ey,maxvalue;
int dir[][]={{,},{,-},{,},{-,}};
struct node
{
int x,y;
int times;
int values;
};
bool judge1(int x,int y)//判断能不能走
{
if(x>=&&x<h&&y>=&&y<w&&g[x][y]!=-)
return true;
return false;
}
bool judge2(int x,int y,int v,int t)//判断应不应该走
{
if(v>value[x][y]) return true;
if(t<time[x][y]) return true;
return false;
}
void bfs()
{
queue<node>q;
node cur,next;
int i;
cur.x=sx;cur.y=sy;cur.times=;cur.values=;
q.push(cur);
while(!q.empty())
{
cur=q.front();
q.pop();
if(cur.x==ex&&cur.y==ey&&cur.times<=l)
{
if(maxvalue<cur.values) maxvalue=cur.values;
}
for(i=;i<;i++)
{
next.x=cur.x+dir[i][];
next.y=cur.y+dir[i][];
if(judge1(next.x,next.y))
{
next.times=cur.times+;
next.values=cur.values+g[next.x][next.y];
if(judge2(next.x,next.y,next.values,next.times))
{
q.push(next);
time[next.x][next.y]=next.times;
value[next.x][next.y]=next.values;
g[next.x][next.y]=;
}
}
}
}
}
int main()
{
int i,j,t,cou=;;
char ch;
cin>>t;
while(t--)
{
cou++;
cin>>w>>h>>l>>m;
for(i=;i<m;i++) cin>>M[i];
getchar();
for(i=;i<h;i++)
{
for(j=;j<w;j++)
{
cin>>ch;
switch(ch)
{
case '*':g[i][j]=-;break;
case '.':g[i][j]=;break;
case '@':g[i][j]=;sx=i;sy=j;break;
case '<':g[i][j]=;ex=i;ey=j;break;
default: g[i][j]=M[ch-''-];break;
}
}
}
memset(value,,sizeof(value));
for(i=;i<h;i++)
for(j=;j<w;j++) time[i][j]=;
maxvalue=-;
bfs();
cout<<"Case "<<cou<<":"<<endl;
if(maxvalue>=)
cout<<"The best score is "<<maxvalue<<"."<<endl<<endl;
else cout<<"Impossible"<<endl<<endl;
}
return ;
}

在网上搜了一下别人的代码,大致就是先利用bfs构建一个隐氏图,然后再用dfs进行搜索,找到最大价值总和。

#include"iostream"
#include"stdio.h"
#include"algorithm"
#include"cmath"
#include"string.h"
#include"string"
#include"queue"
#define mx 105
using namespace std;
char mp[mx][mx];
bool used[mx][mx];//标记bfs走过的路径
bool vis[mx];//标记 dfs走过的路径
int value[mx];//记录宝物的价值
int H,W,L,M,ans,sum;
int step[mx][mx];//记录每个位置的最小步骤
int dis[mx][mx];//记录出口、入口、宝物两两之间的最短距离
queue<int>q;
int dir[][]={{,},{,-},{,},{-,}};
bool judge(int x,int y)//判断该位置是否可走
{
if(x>=&&x<H&&y>=&&y<W&&mp[x][y]!='*') return true;
return false;
}
void bfs(int x1,int y1,int s)
{
while(!q.empty()) q.pop();
int dx,dy,i,u,x,y;
memset(used,false,sizeof(used));
memset(step,,sizeof(step));
u=x1*W+y1;
q.push(u);
used[x1][y1]=true;
step[x1][y1]=;
while(!q.empty())
{
u=q.front();
q.pop();
x=u/W;
y=u%W;
for(i=;i<;i++)
{
dx=x+dir[i][];
dy=y+dir[i][];
if(judge(dx,dy))
{
if(!used[dx][dy])
{
used[dx][dy]=true;
step[dx][dy]=step[x][y]+;
if(mp[dx][dy]=='@')
dis[s][]=step[dx][dy];
else if(mp[dx][dy]=='<')
dis[s][M+]=step[dx][dy];
else if(mp[dx][dy]>='A'&&mp[dx][dy]<='J')
dis[s][mp[dx][dy]-'A'+]=step[dx][dy];
q.push(dx*W+dy);
}
}
}
}
}
void dfs(int s,int val,int time)
{
int i;
if(time>L) return;
if(ans==sum) return;
if(s>M)
{
if(val>ans)ans=val;
return;
}
for(i=;i<=M+;i++)
{
if(dis[s][i]==||vis[i]) continue;
vis[i]=true;
dfs(i,value[i]+val,time+dis[s][i]);
vis[i]=false;
}
}
int main()
{
int i,j,t,icase=;
cin>>t;
while(t--)
{
sum=;ans=-;
memset(dis,,sizeof(dis));//记得初始化dis
icase++;
cin>>W>>H>>L>>M;
for(i=;i<=M;i++) {cin>>value[i];sum+=value[i];}
for(i=;i<H;i++) cin>>mp[i];
value[]=;value[M+]=;
for(i=;i<H;i++)
{
for(j=;j<W;j++)
{
if(mp[i][j]=='@') bfs(i,j,);
else if(mp[i][j]=='<')bfs(i,j,M+);
else if(mp[i][j]>='A'&&mp[i][j]<='J') bfs(i,j,mp[i][j]-'A'+);
}
}
memset(vis,false,sizeof(vis));
vis[]=true;
dfs(,,);
cout<<"Case "<<icase<<":"<<endl;
if(ans>)
cout<<"The best score is "<<ans<<"."<<endl;
else cout<<"Impossible"<<endl;
if(t) cout<<endl;
}
}

hdu Collect More Jewels的更多相关文章

  1. HDU Collect More Jewels 1044

    BFS + 状态压缩 险过 这个并不是最好的算法 但是写起来比较简单 , 可以AC,但是耗时比较多 下面是代码 就不多说了 #include <cstdio> #include <c ...

  2. HDU 1044 Collect More Jewels(BFS+DFS)

    Collect More Jewels Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Othe ...

  3. hdu.1044.Collect More Jewels(bfs + 状态压缩)

    Collect More Jewels Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Othe ...

  4. hdu 1044 Collect More Jewels(bfs+状态压缩)

    Collect More Jewels Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Othe ...

  5. Collect More Jewels(hdu1044)(BFS+DFS)

    Collect More Jewels Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Othe ...

  6. hdu 1044 Collect More Jewels

    题意: 一个n*m的迷宫,在t时刻后就会坍塌,问:在逃出来的前提下,能带出来多少价值的宝藏. 其中: ’*‘:代表墙壁: '.':代表道路: '@':代表起始位置: '<':代表出口: 'A'~ ...

  7. hdu 1044(bfs+状压)

    非常经典的一类题型 没有多个出口.这里题目没有说清楚 Collect More Jewels Time Limit: 2000/1000 MS (Java/Others)    Memory Limi ...

  8. HDU 1044 BFS

    Collect More Jewels Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Othe ...

  9. hdu 1044(bfs+dfs+剪枝)

    Collect More Jewels Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Othe ...

随机推荐

  1. host

    #Google Services START209.116.186.241 0.docs.google.com209.116.186.241 0.drive.google.com209.116.186 ...

  2. 31.从尾到头输出链表[Print linked list from last to first]

    [题目] 输入一个链表的头结点,从尾到头反过来输出每个结点的值. [分析] 这是一道很有意思的面试题.该题以及它的变体经常出现在各大公司的面试.笔试题中. [链表逆置] 看到这道题后,第一反应是从头到 ...

  3. eclipse的c++工程开启c++11

    右击工程->Properties->C/C++ Builder->Setting->Tool Setting->Miscellanous->Other Flags添 ...

  4. iOS 推荐学习__bridge等ARC知识的好资料

    请下载 iOS5 by Tutorials!写得很好的!

  5. (原创)Python文件与文件系统系列(5)——stat模块

    stat模块中定义了许多的常量和函数,可以帮助解释 os.stat().os.fstat().os.lstat()等函数返回的 st_result 类型的对象. 通常使用 os.path.is*() ...

  6. 1.前端笔记之html

    title: 1.前端笔记之HTML date: 2016-04-04 23:21:52 tags: Python categories: Python --- 作者:刘耀 **出处:http://w ...

  7. Java性能优化权威指南-读书笔记(四)-JVM性能调优-延迟

    延迟指服务器处理一个请求所花费的时间,单位一般是ms.s. 本文主要讲降低延迟可以做的服务器端JVM优化. JVM延迟优化 新生代 新生代大小决定了应用平均延迟 如果平均Minor GC持续时间大于应 ...

  8. 泥泞的道路(codevs 1183)

    题目描述 Description CS有n个小区,并且任意小区之间都有两条单向道路(a到b,b到a)相连.因为最近下了很多暴雨,很多道路都被淹了,不同的道路泥泞程度不同.小A经过对近期天气和地形的科学 ...

  9. CocoaPods报错及解决方法记录

    [!] Oh no, an error occurred. Search for existing GitHub issues similar to yours: https://github.com ...

  10. sina发现并不会去导入qq使用的

    看问题需要多角度,为之不能实现也是有可能没有完善的.确实是由于短时间发布过多,还是bky好点好像有30S