正统双端队列搜索

回顾:普通队列进行边权为定值的最短路

每次到达都是最优的(意味着不用取min

why?

因为所有状态按照 入队的先后顺序 具有 层次单调性,每次扩展,都往外走一步,满足从起始到该状态的最优性(不用取min/也不用比大小,如果如此失去了意义)

回到正题:双端队列可以进行边权两个定值(我们在此简化成1/0)的最短路

操作:对于一条边u到v,如果此边权值为0,我们将它push_front(v),否则push_back(k),每次取队首

这样我们保证了单调性(即每次优先选择最优的)

注意细节:我们放入队列里的还有u,这样才能做到将每次取出的时候更新,也就是说在队列中放入二元组(u,v),具体看代码

至于建边(其实我没有建),只要认为某一方格对角两点间有边,边权为0,否则为1

复杂度:O(r*c)

实测:开o2(76ms)(毕竟deque常数大)

不开o2(300ms)速度相当可以

希望大家能通过此题正确认识到双端队列的bfs(不用取min哦)

#include<iostream>
#include<cstdio>
#include<cstring>
#include<deque>
using namespace std;
const int MAXX=550;
int dis[MAXX][MAXX];
bool vis[MAXX][MAXX],Map[MAXX][MAXX];
int dx[4]={1,-1,1,-1};
int dy[4]={1,-1,-1,1};//此处为了方便
int t,r,c;
char s[MAXX];
inline void init(){
memset(dis,0,sizeof(dis));
memset(vis,0,sizeof(vis));
}
inline bool check(int x,int y){
if(x>=1&&y>=1&&x<=r+1&&y<=c+1)return true;
else return false;
}//检查是否到边界
inline int edge(int x,int xx,int y,int yy){
if((xx<x&&yy<y)||(xx>x&&yy>y))return !Map[min(x,xx)][min(y,yy)];
else return Map[min(x,xx)][min(y,yy)];
}
inline void bfs(){
Map[0][0]=1;
deque< pair < pair<int ,int> ,pair<int ,int > > >q;//二元组队列
pair<int ,int > u=make_pair(0,0);
pair<int ,int > v=make_pair(1,1);
q.push_back(make_pair(u,v));
dis[0][0]=0;
while(!q.empty()){
pair< pair<int ,int > , pair<int ,int > >t=q.front();
q.pop_front();
pair<int ,int >u=t.first;
pair<int ,int >v=t.second;
int xx=u.first;
int yy=u.second;
int x=v.first;
int y=v.second;
if(vis[x][y])continue;//这里保证复杂度O(r*c)
dis[x][y]=dis[xx][yy]+edge(x,xx,y,yy);
vis[x][y]=1;
for(int i=0;i<4;++i){
int xv=x+dx[i];
int yv=y+dy[i];
pair<int ,int >s=make_pair(xv,yv);
if(check(xv,yv)&&!vis[xv][yv]){
if(edge(xv,x,yv,y))q.push_back(make_pair(v,s));
else q.push_front(make_pair(v,s));
}
}
}
}
int main(){
scanf("%d",&t);
while(t--){
init();
scanf("%d%d",&r,&c);
for(int i=1;i<=r;++i){
scanf("%s",s+1);
for(int j=1;j<=c;++j)
if(s[j]=='\\')Map[i][j]=1;//这里有c++的转义字符我么可以这么处理/
else Map[i][j]=0;
}
bfs();
if(vis[r+1][c+1])printf("%d\n",dis[r+1][c+1]);
else printf("NO SOLUTION\n");
}
return 0;
}

[lougu2243]双端队列搜索的更多相关文章

  1. 06.队列、python标准库中的双端队列、迷宫问题

    class QueueUnderflow(ValueError): """队列为空""" pass class SQueue: def __ ...

  2. 用python实现栈/队列/双端队列/链表

    栈是元素的有序集合,添加操作与移除操作都发生在其顶端,先进后出栈操作:创建空栈,增删(顶端),查(顶端元素,元素个数,是否为空)应用:将十进制数转换成任意进制数 class Stack: # 用列表创 ...

  3. STL-Deque(双端队列)与单调队列的实现

    前言: STl是个好东西,虽然他在不开O2的条件下会跑的很慢,但他着实会让你的代码可读性大大提高,令你的代码看起来既简单又整洁. 双端队列: 顾名思义,双端队列是有两个头的,一个队首指针,一个队尾指针 ...

  4. 队列(Queue)\双端队列(Deque)

    队列(Queue)\双端队列(Deque) 队列(Queue) 双端队列(Deque) 算法应用 队列(Queue) 特点: 和栈不同,队列的最大特点是先进先出(FIFO),就好像按顺序排队一样.对于 ...

  5. 电路维修(双端队列 & 最短路)

    达达是来自异世界的魔女,她在漫无目的地四处漂流的时候,遇到了善良的少女翰翰,从而被收留在地球上. 翰翰的家里有一辆飞行车. 有一天飞行车的电路板突然出现了故障,导致无法启动. 电路板的整体结构是一个$ ...

  6. lintcode二叉树的锯齿形层次遍历 (双端队列)

    题目链接: http://www.lintcode.com/zh-cn/problem/binary-tree-zigzag-level-order-traversal/ 二叉树的锯齿形层次遍历 给出 ...

  7. lintcode 滑动窗口的最大值(双端队列)

    题目链接:http://www.lintcode.com/zh-cn/problem/sliding-window-maximum/# 滑动窗口的最大值 给出一个可能包含重复的整数数组,和一个大小为  ...

  8. STL---deque(双端队列)

    Deque是一种优化了的.对序列两端元素进行添加和删除操作的基本序列容器.它允许较为快速地随机访问,但它不像vector 把所有的对象保存在一块连续的内存块,而是采用多个连续的存储块,并且在一个映射结 ...

  9. hdu-5929 Basic Data Structure(双端队列+模拟)

    题目链接: Basic Data Structure Time Limit: 7000/3500 MS (Java/Others)    Memory Limit: 65536/65536 K (Ja ...

随机推荐

  1. ReactiveCocoa结合了几种编程风格

    函数式编程(Functional Programming):使用高阶函数,例如函数用其他函数作为参数.响应式编程(Reactive Programming):关注于数据流和变化传播.所以,你可能听说过 ...

  2. 关于common.js里面的module.exports与es6的export default的思考总结

    背景 公司项目需要裁切功能,基于第三方图片裁切组件vue-cropper(0.4.0版本),封装了图片裁切组件(picture-cut)(放在公司内部组件库,仅限于公司内部使用) 在vue-cropp ...

  3. mysql更改密码与远程管理

    set password = ': #在当前用户下更改密码 grant all privileges on *.* to root@"%" identified by " ...

  4. 洛谷 P1293 班级聚会

    P1293 班级聚会 题目描述 毕业25年以后,我们的主人公开始准备同学聚会.打了无数电话后他终于搞到了所有同学的地址.他们有些人仍在本城市,但大多数人分散在其他的城市.不过,他发现一个巧合,所有地址 ...

  5. Java Exception和Error的差别

    Java中异常的抽象类是Throwable,在此基础上.派生出两大类:Error和Exception. Error是程序中的严重错误,不应该用try-catch包括.Javadoc的说明例如以下: A ...

  6. 学习笔记 Java_静态_继承 2014.7.12

    一.静态 1.构造函数:       特点:       1. 函数名和类名同样.       2. 不用定义返回值类型(和void不是一回事,而构造函数是根本不用定义返回值类型).       3. ...

  7. 找出一堆数中最小的前K个数

    描写叙述: 给定一个整数数组.让你从该数组中找出最小的K个数 思路: 最简洁粗暴的方法就是将该数组进行排序,然后取最前面的K个数就可以. 可是,本题要求的仅仅是求出最小的k个数就可以,用排序能够但显然 ...

  8. Ubuntu系统下的Mysql安装与使用

    摘要 在本篇博文中.笔者将从基础出发.介绍Mysql在Linux环境下的安装和基本使用命令,仅适用于Mysql刚開始学习的人.大牛请绕道-- 安装Mysql数据库 这里介绍最最简单的安装方式,至于编译 ...

  9. DevExpress控件的安装及画图控件的使用

    近期须要绘制纵断面图,而AE自带的又不是非常好,查找资料后使用DevExpress控件中的画图控件实现了纵断面的绘制.Dev控件是须要付费的.这里我们使用破解版的哈. 安装包及破解文件上传至我的网盘了 ...

  10. Install the IIS 6.0 Management Compatibility Components in Windows 7 or in Windows Vista from Control Panel

    https://technet.microsoft.com/en-us/library/bb397374(v=exchg.80).aspx Install the IIS 6.0 Management ...