题意:

             给你起点,终点,图上有墙有路还有宝物,问你在规定时间内能否能到终点,如果能问最多能捡到多少宝物.


思路:

          看完这个题目果断 BFS+三维的mark[][][] 第三维用二进制压缩的思想去解决,结果TLE了,我后来在网上看了看,发现有人用二进制压缩ac了,这更坚定了我的决心啊,不停的优化 超时 优化 超时 就这样我超时了 50多次,我tm恶心了,直接粘了个网上说和我想法一样的代码交上去了,结果 TLE 了, 我 fuck ,超时了早说啊... ,可能是数据加强了.

下面说一下正解 ,观察我们发现,题目中给的宝贝数量并不多,最多才10 ,其实我们可以先 BFS求出所有点的最短路,这些点包括 ,起点,终点,宝物,求完最短路我们其实就把这个图压缩成了最多 10 + 1 + 1 个点,压缩了图之后就是暴力DFS了,12个点压力也有点大,所以要剪枝 , 在深搜的时候如果发现当前的可满足ans 已经等于宝物总和了,那么不用在搜了,直接return 到最外面,或者可以再加上一个就是,当前的所用时间 + 当前到终点的时间
大于题目给的时间限制了,直接return 到上一层; 



/*

先BFS把图压缩,在DFS爆搜

*/

#include<stdio.h>

#include<string.h>

#include<queue>

#define N1 60

#define N2 15 

using namespace std;

typedef struct

{

   int x ,y ,t;

}NODE;

NODE xin ,tou;

int mark_bfs[N1][N1] ,mark_dfs[N2];

int dis[N2][N2] ,money[N2];

int dir[4][2] = {0 ,1 ,0 ,-1 ,1 ,0 ,-1 ,0};

char map[N1][N1];

int xx ,yy ,tt ,mm ,ss ,nn;

bool ok(int x ,int y)

{

   if(x >= 0 && x < xx && y >= 0 && y < yy && map[x][y] != '*' && !mark_bfs[x][y])

   return 1;

   return 0;

}

void BFS(int x ,int y ,int key)// 求那几个点之间的最短路,key当前是起始点.

{

   memset(mark_bfs ,0 ,sizeof(mark_bfs));

   mark_bfs[x][y] = 1;

   xin.x = x ,xin.y = y ,xin.t = 0;

   queue<NODE>q;

   q.push(xin);

   while(!q.empty())

   {

      tou = q.front();

      q.pop();

      for(int i = 0 ;i < 4 ;i ++)

      {

         xin.x = tou.x + dir[i][0];

         xin.y = tou.y + dir[i][1];

         xin.t = tou.t + 1;

         if(ok(xin.x ,xin.y))

         {

            mark_bfs[xin.x][xin.y] = 1;

            q.push(xin);

            if(map[xin.x][xin.y] == '@')

            dis[key][0] = xin.t;

            if(map[xin.x][xin.y] == '<')

            dis[key][nn+1] = xin.t;

            if(map[xin.x][xin.y] >= 'A' && map[xin.x][xin.y] <= 'J')

            dis[key][map[xin.x][xin.y] - 64] = xin.t;

         }

      }

   }

   return ;

}

void DFS(int s ,int time ,int mon) // s是当前点 ,time是当前所用时间,mon是当前财

//富值

{

   if(time > tt || mm == ss) return;

   if(s == nn + 1 && mm < mon)

   mm = mon;

   for(int k = 1 ;k <= nn + 1 ;k ++)

   {

      if(!dis[s][k] || mark_dfs[k]) continue;

      mark_dfs[k] = 1;

      DFS(k ,time + dis[s][k] ,mon + money[k]);

      mark_dfs[k] = 0;

   }

}

int main ()

{

   int i ,j ,t ,cas = 1;

   scanf("%d" ,&t);

   while(t--)

   {

      scanf("%d %d %d %d" ,&yy ,&xx ,&tt ,&nn);

      money[0] = money[nn+1] = 0;

      for(ss = 0 ,i = 1 ;i <= nn ;i ++)

      {

         scanf("%d" ,&money[i]);

         ss += money[i];

      }

      for(i = 0 ;i < xx ;i ++)

      scanf("%s" ,map[i]);

      memset(dis ,0 ,sizeof(dis));

      for(i = 0 ;i < xx ;i ++)

      for(j = 0 ;j < yy ;j ++)

      {

         // 起点是 0 ,终点是 n + 1 ,宝物是 1 ---- n其他的墙和路都不要了,深搜时没有用

         if(map[i][j] == '*')

         continue;

         if(map[i][j] == '@')

         BFS(i ,j ,0);

         if(map[i][j] == '<')

         BFS(i ,j ,nn + 1);

         if(map[i][j] >= 'A' && map[i][j] <= 'J')

         BFS(i ,j ,map[i][j] - 64);

      }

      mm = -1;

      memset(mark_dfs ,0 ,sizeof(mark_dfs));

      mark_dfs[0] = 1;

      DFS(0 ,0 ,0);

      printf("Case %d:\n" ,cas ++);

      if(mm == -1)

      printf("Impossible\n");

      else

      printf("The best score is %d.\n" ,mm);

      if(t) printf("\n");

   }

   return 0;

}

   

      

      

hdu 1044 BFS(压缩图)+DFS的更多相关文章

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

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

  2. HDU 1983 BFS&amp;&amp;DFS

    大多数刚需封锁4区域可以,DFS地区封锁.BFS无论是通过 #include "stdio.h" #include "string.h" #include &q ...

  3. hdu 1044(bfs+状压)

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

  4. HDU 1044 BFS

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

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

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

  6. 列出连通集(DFS及BFS遍历图) -- 数据结构

    题目: 7-1 列出连通集 (30 分) 给定一个有N个顶点和E条边的无向图,请用DFS和BFS分别列出其所有的连通集.假设顶点从0到N−1编号.进行搜索时,假设我们总是从编号最小的顶点出发,按编号递 ...

  7. CF467D Fedor and Essay 建图DFS

      Codeforces Round #267 (Div. 2) CF#267D D - Fedor and Essay D. Fedor and Essay time limit per test ...

  8. HDU 1241 Oil Deposits --- 入门DFS

    HDU 1241 题目大意:给定一块油田,求其连通块的数目.上下左右斜对角相邻的@属于同一个连通块. 解题思路:对每一个@进行dfs遍历并标记访问状态,一次dfs可以访问一个连通块,最后统计数量. / ...

  9. hdu 1241 Oil Deposits(DFS求连通块)

    HDU 1241  Oil Deposits L -DFS Time Limit:1000MS     Memory Limit:10000KB     64bit IO Format:%I64d & ...

随机推荐

  1. Java 学习阶段性感想

    阶段性感想·操千曲而后晓声 回顾 从2月17日 到 今天 4月19日,我算是暂时完成了Java入门的学习了. 从基本语法到面向对象,从常见API到字符串集合,从文件处理到多线程,我学到了很多,很多很多 ...

  2. 【MaixPy3文档】写好 Python 代码!

    本文是给有一点 Python 基础但还想进一步深入的同学,有经验的开发者建议跳过. 前言 上文讲述了如何认识开源项目和一些编程方法的介绍,这节主要来说说 Python 代码怎么写的一些演化过程和可以如 ...

  3. CVE-2016-10033 WordPress <= 4.6 命令执行漏洞

    漏洞参考 https://www.jianshu.com/p/85ac4af9f947 漏洞信息 这个锅还是要PHPMailer背(CVE-2016-10033,WordPress 使用 PHPMai ...

  4. Java 树结构实际应用 二(哈夫曼树和哈夫曼编码)

     赫夫曼树 1 基本介绍 1) 给定 n 个权值作为 n 个叶子结点,构造一棵二叉树,若该树的带权路径长度(wpl)达到最小,称这样的二叉树为 最优二叉树,也称为哈夫曼树(Huffman Tree), ...

  5. C语言之漫谈指针(上)

    C语言之漫谈指针(上) 在C语言学习的途中,我们永远有一个绕不了的坑,那就是--指针. 在这篇文章中我们就谈一谈指针的一些基础知识. 纲要: 零.谈指针之前的小知识 一.指针与指针变量 二.指针变量的 ...

  6. 2019看雪CTF 晋级赛Q2第四题wp

    上次参加2019看雪CTF 晋级赛Q2卡在了这道题上,虽然逆出算法,但是方程不会解,哈哈哈哈,果然数学知识很重要呀,现在记录一下. 首先根据关键信息,根据错误提示字符串定位到这里: 1 int __t ...

  7. Kubernetes 实战 —— 05. 服务:让客户端发现 pod 并与之通信(下)

    将服务暴露给外部客户端 P136 有以下三种方式可以在外部访问服务: 将服务的类型设置成 NodePort 将服务的类型设置为 LoadBalance 创建一个 Ingress 资源 使用 NodeP ...

  8. 结对作业-stage_2

    见队友博客:结对编程-stage_2

  9. JavaWeb 补充(JSP&EL&JSTL)

    1. JSP:     1. 指令     2. 注释     3. 内置对象 2. MVC开发模式 3. EL表达式 4. JSTL标签 5. 三层架构 JSP: 1. 指令     * 作用:用于 ...

  10. 6.2set用法

    目录 1.set的定义 2.set容器内元素的访问 3.set常见使用的函数 set可以内部进行自动递增排序,且自动去除了重复元素 1.set的定义 set<typename> name; ...