热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

poj3278简单的广搜入门

这道题是一道最基础的广度优先搜索题,需要剪枝,否则超时,剪枝时剪掉已经计算过的数,否则会式计算次数成幂指数增长。广度搜索最基本的思想就是遍历每一层,之后遍历下一层,因此剪枝十分重要。我开始考虑用

这道题是一道最基础的广度优先搜索题,需要剪枝,否则超时,剪枝时剪掉已经计算过的数,否则会式计算次数成幂指数增长。广度搜索最基本的思想就是遍历每一层,之后遍历下一层,因此剪枝十分重要。我开始考虑用数组来存放到达y所需步数a[y],后发现用队列实现更好,由于队列先入先出的特点,所以使用了stl中的队列实现。

stl中队列基本用法:

头文件:

#include
基本函数:

建立队列:queue+队列名;

向队列中(末尾)添加元素:队列名+.push(元素名);

去掉队列头元素:队列名+.pop();

队列头元素:队列名+.front();

返回队尾元素的引用:队列名+.back();

判断是否为空:队列名+.empty();

队列大小:队列名+.size();

代码:

#include
#include
using namespace std;
int a[100001],b[100001];
int main()
{
int m,n,y;
queuex;//建立名为x的队列
cin>>m>>n;
x.push(m);//在队列后加入m
a[m]=0;
while(x.size()!=0)//队列大小不为0
{
  y=x.front();//y取队列首位
  x.pop();//去掉队列首位
  b[y]=1;
  if(y==n)
   break;
  if(y-1>=0&&b[y-1]==0)
  {
   x.push(y-1);
   a[y-1]=a[y]+1;
   b[y-1]=1;
  }
  if(y+1<=100000&&b[y+1]==0)
  {
   x.push(y+1);
   a[y+1]=a[y]+1;
   b[y+1]=1;
  }
  if(2*y<=100000&&b[2*y]==0)
  {
   x.push(2*y);
   a[2*y]=a[y]+1;
   b[2*y]=1;
  }
}
cout< return 0;
}

上面是别人的代码,写的很好。下面是我自己仿写的代码和找的一篇不用stl 的更高效代码

这是自己写的  memory 1196K    time   141Ms

#include
#include
#include
#include
using namespace std;
int a[100001],v[100001];
int main()
{
int n,m,u,i=0,j;
memset(v,0,sizeof(v));
queue x;
scanf("%d%d",&n,&m);
x.push(n);
a[n]=0;
while(!x.empty())
{
  u=x.front();
  x.pop();
  v[u]=1;
  if(u==m)
   break;
  if(u-1>=0&&v[u-1]==0)
  {
   x.push(u-1);
   v[u-1]=1;
   a[u-1]=a[u]+1;
  }
  if(u+1<=100000&&v[u+1]==0)
  {
   x.push(u+1);
   v[u+1]=1;
   a[u+1]=a[u]+1;
  }
  if(2*u<=100000&&v[2*u]==0)
  {
   x.push(2*u);
   v[2*u]=1;
   a[2*u]=a[u]+1;
  }
}
printf("%d\n",a[m]);
return 0;
}

SOU DI不用stl 的算法

memory   980K  time 16Ms

#include   
#include   
#include   
#include   
#include   
#include   
#include   
#include   
using namespace std;  
bool data[100005] = {0};  
int que[100005] = {0};  
int tnum[100005] = {0};  
void bfs(int n, int k)  
{  
   int frOnt= 0, rear = 0;  
    que[0] = n;  
    data[n] = true;  
    tnum[0] = 0;  
    rear++;  
      
    while (front != rear)  
    {  
        int t = que[front];  
        int tn = tnum[front];  
        if (t == k)  
        {  
            printf("%d\n", tn);  
            return;  
        }  
          
        if (t > 0 && !data[t-1])  
        {  
            data[t-1] = true;  
            que[rear] = t-1;  
            tnum[rear] = tn+1;  
            rear++;  
        }  
          
        if (t <100000 && !data[t+1])  
        {  
            data[t+1] = true;  
            que[rear] = t+1;  
            tnum[rear] = tn+1;  
            rear++;  
        }  
        if (t <= 50000 && !data[t*2])  
        {  
            data[t*2] = true;  
            que[rear] = t*2;  
            tnum[rear] = tn+1;  
            rear++;  
        }  
        front++;  
    }  
    printf("0\n");  
}  
      
int main()  
{  
    int n, k;  
    scanf("%d%d", &n, &k);  
    memset(data, 0, sizeof(data));  
    bfs(n, k);  
    return 0;  


推荐阅读
  • 李逍遥寻找仙药的迷阵之旅
    本文讲述了少年李逍遥为了救治婶婶的病情,前往仙灵岛寻找仙药的故事。他需要穿越一个由M×N个方格组成的迷阵,有些方格内有怪物,有些方格是安全的。李逍遥需要避开有怪物的方格,并经过最少的方格,找到仙药。在寻找的过程中,他还会遇到神秘人物。本文提供了一个迷阵样例及李逍遥找到仙药的路线。 ... [详细]
  • 向QTextEdit拖放文件的方法及实现步骤
    本文介绍了在使用QTextEdit时如何实现拖放文件的功能,包括相关的方法和实现步骤。通过重写dragEnterEvent和dropEvent函数,并结合QMimeData和QUrl等类,可以轻松实现向QTextEdit拖放文件的功能。详细的代码实现和说明可以参考本文提供的示例代码。 ... [详细]
  • 本文讨论了使用差分约束系统求解House Man跳跃问题的思路与方法。给定一组不同高度,要求从最低点跳跃到最高点,每次跳跃的距离不超过D,并且不能改变给定的顺序。通过建立差分约束系统,将问题转化为图的建立和查询距离的问题。文章详细介绍了建立约束条件的方法,并使用SPFA算法判环并输出结果。同时还讨论了建边方向和跳跃顺序的关系。 ... [详细]
  • JVM 学习总结(三)——对象存活判定算法的两种实现
    本文介绍了垃圾收集器在回收堆内存前确定对象存活的两种算法:引用计数算法和可达性分析算法。引用计数算法通过计数器判定对象是否存活,虽然简单高效,但无法解决循环引用的问题;可达性分析算法通过判断对象是否可达来确定存活对象,是主流的Java虚拟机内存管理算法。 ... [详细]
  • 本文介绍了解决二叉树层序创建问题的方法。通过使用队列结构体和二叉树结构体,实现了入队和出队操作,并提供了判断队列是否为空的函数。详细介绍了解决该问题的步骤和流程。 ... [详细]
  • 开发笔记:实验7的文件读写操作
    本文介绍了使用C++的ofstream和ifstream类进行文件读写操作的方法,包括创建文件、写入文件和读取文件的过程。同时还介绍了如何判断文件是否成功打开和关闭文件的方法。通过本文的学习,读者可以了解如何在C++中进行文件读写操作。 ... [详细]
  • 深入理解Kafka服务端请求队列中请求的处理
    本文深入分析了Kafka服务端请求队列中请求的处理过程,详细介绍了请求的封装和放入请求队列的过程,以及处理请求的线程池的创建和容量设置。通过场景分析、图示说明和源码分析,帮助读者更好地理解Kafka服务端的工作原理。 ... [详细]
  • 本文介绍了Codeforces Round #321 (Div. 2)比赛中的问题Kefa and Dishes,通过状压和spfa算法解决了这个问题。给定一个有向图,求在不超过m步的情况下,能获得的最大权值和。点不能重复走。文章详细介绍了问题的题意、解题思路和代码实现。 ... [详细]
  • 本文介绍了在Android开发中使用软引用和弱引用的应用。如果一个对象只具有软引用,那么只有在内存不够的情况下才会被回收,可以用来实现内存敏感的高速缓存;而如果一个对象只具有弱引用,不管内存是否足够,都会被垃圾回收器回收。软引用和弱引用还可以与引用队列联合使用,当被引用的对象被回收时,会将引用加入到关联的引用队列中。软引用和弱引用的根本区别在于生命周期的长短,弱引用的对象可能随时被回收,而软引用的对象只有在内存不够时才会被回收。 ... [详细]
  • STL迭代器的种类及其功能介绍
    本文介绍了标准模板库(STL)定义的五种迭代器的种类和功能。通过图表展示了这几种迭代器之间的关系,并详细描述了各个迭代器的功能和使用方法。其中,输入迭代器用于从容器中读取元素,输出迭代器用于向容器中写入元素,正向迭代器是输入迭代器和输出迭代器的组合。本文的目的是帮助读者更好地理解STL迭代器的使用方法和特点。 ... [详细]
  • VScode格式化文档换行或不换行的设置方法
    本文介绍了在VScode中设置格式化文档换行或不换行的方法,包括使用插件和修改settings.json文件的内容。详细步骤为:找到settings.json文件,将其中的代码替换为指定的代码。 ... [详细]
  • 3.223.28周学习总结中的贪心作业收获及困惑
    本文是对3.223.28周学习总结中的贪心作业进行总结,作者在解题过程中参考了他人的代码,但前提是要先理解题目并有解题思路。作者分享了自己在贪心作业中的收获,同时提到了一道让他困惑的题目,即input details部分引发的疑惑。 ... [详细]
  • 本文介绍了最长上升子序列问题的一个变种解法,通过记录拐点的位置,将问题拆分为左右两个LIS问题。详细讲解了算法的实现过程,并给出了相应的代码。 ... [详细]
  • Android工程师面试准备及设计模式使用场景
    本文介绍了Android工程师面试准备的经验,包括面试流程和重点准备内容。同时,还介绍了建造者模式的使用场景,以及在Android开发中的具体应用。 ... [详细]
  • 重入锁(ReentrantLock)学习及实现原理
    本文介绍了重入锁(ReentrantLock)的学习及实现原理。在学习synchronized的基础上,重入锁提供了更多的灵活性和功能。文章详细介绍了重入锁的特性、使用方法和实现原理,并提供了类图和测试代码供读者参考。重入锁支持重入和公平与非公平两种实现方式,通过对比和分析,读者可以更好地理解和应用重入锁。 ... [详细]
author-avatar
超人
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有