题解-[HNOI2001]遥控赛车比赛
题解-[HNOI2001]遥控赛车比赛
前置知识:记忆化搜索、\(\texttt{Bfs}\)。
参考资料
跳转按钮
题解-[HNOI2001]遥控赛车比赛 |
---|
\(\texttt{Introduction}\) |
\(\texttt{Description}\) |
\(\texttt{Solution}\) |
\(\texttt{Code}\) |
\(\texttt{Introduction}\)
蒟蒻练习历年省选题时遇见此题,\(\texttt{WA}\) 了好多发才 \(\texttt{AC}\),感到这题的巧妙足以记成题解。
\(\texttt{Description}\)
给你一个由 \(0\) 和 \(1\) 组成的 \(N\times M\) 地图,\(1\) 可走,\(0\) 是障碍。如果你反应力为 \(z\),那么你每次至少直走 \(z\) 步后才能转弯。起点为 \((sx,sy)\),终点为 \((tx,ty)\)。求反应力为 \(1\sim 10\) 时的最短路(如果到不了终点不输出,具体看题目链接)。
数据范围:\(1\le N,M\le 100\)。
\(\texttt{Solution}\)
这题是什么意思呢?如下图:
起点为 \((1,4)\),终点为 \((10,7)\)。
如果反应力为 \(1\),即走一步可以转个方向,那么最短路方案如图 \(2\),长度为 \(16\)。
如果反应力为 \(2\),即每直走两步可以换个方向,那么最短路方案如图 \(1\),长度为 \(18\)。
貌似很简单,做法很直接:\(\texttt{Bfs}\),记忆化搜索。
因为方向在这题中很重要,所以记录数组 \(dep_{i,j,k}\) 和 \(f_{i,j,k}\) 表示走到 \((i,j)\) 这个格子方向为 \(k\) 时在最短路条件下的路程和当前方向上直走了的距离(四个方向用 \(0,1,2,3\) 表示)。
然后每次 \(\texttt{Bfs}\) 拓展下一步路径的时候,特判一下,如果方向与 \(k\) 不同,那么必须满足 \(f_{i,j,k}\ge z\)。
看起来这题就只有普及的复杂度,但是如果你把代码敲出来一交,最多得个 \(50\) 分。
你会自闭很久直到找到反例——一个更令你自闭的东西:
路重复走,转 \(180^{\circ}\) 反而可能更优。
如下图:
图 \(1\) 中的地图如果用你的代码只能跑反应力为 \(1\)。但是这种路径反应力为 \(2\) 可以跑!
同理,图 \(2\) 中如果走这种路重复走,转 \(180^{\circ}\) 的路径,长度可以减至 \(10\)(虚线路径为原计划路径,长度远大于 \(10\))。
那么题目貌似会变得很混乱:走过的地方也可以走,同一个位置更长的路径可能更优。
然而仔细思考后会发现,只有两种情况同一个位置的路径会更优:
更短。
当前方向上直走得更长。
其中第二种更优只能在继续直走中体现(如果转弯之前直走了多长没关系)。
然后敲个带个特判的记忆化广搜即可(普及知识不多说,特判会在代码中标识)。
\(\texttt{Code}\)
#include <bits/stdc++.h>
using namespace std;
//&Start
#define lng long long
#define lit long double
#define kk(i,n) "\n "[i<n]
const int inf=0x3f3f3f3f;
const lng Inf=1e17;
//&Debug
int open(0);
#define Debug if(open)
//&Data
const int N=110;
int n,m,maze[N][N],fx[4]={0,0,-1,1},fy[4]={-1,1,0,0};
struct Marco{int x,y,k;}s,t; //Bfs 状态
//&Bfs
const int Q=5e6+10;
int dep[N][N][4],f[N][N][4],ql,qr;
Marco q[Q];
int ok(int x,int y){return x>=1&&x<=n&&y>=1&&y<=m&&maze[x][y];}
int bfs(int z){
Debug printf("---%d---\n",z);//###
if(!ok(s.x,s.y)||!ok(t.x,t.y)) return -1;
memset(dep,0x3f,sizeof dep),memset(f,0,sizeof f);
int *D=dep[s.x][s.y],*F=f[s.x][s.y]; Marco tp;
D[0]=D[1]=D[2]=D[3]=F[0]=F[1]=F[2]=F[3]=0;//初始化起点
ql=1,qr=0,q[++qr]=(s.k=0,s),q[++qr]=(s.k=1,s),q[++qr]=(s.k=2,s),q[++qr]=(s.k=3,s);//起点可以是任意方向
while(qr>=ql){ //手模队列
tp=q[ql++];
Debug printf("%d %d %d\n",tp.x,tp.y,tp.k);//###
D=dep[tp.x][tp.y],F=f[tp.x][tp.y];
if(tp.x==t.x&&tp.y==t.y) return D[tp.k];//找到出口了!
for(int k=0;k<4;k++){
int xt=tp.x+fx[k],yt=tp.y+fy[k];
if(ok(xt,yt)&&(D[tp.k]+1<dep[xt][yt][k]||F[tp.k]>=f[xt][yt][k])){//特判:两种更优
if(k==tp.k){
dep[xt][yt][k]=D[tp.k]+1;//普通Bfs拓展
f[xt][yt][k]=F[tp.k]+1;
q[++qr]=(Marco){xt,yt,k};
} else if(F[tp.k]>=z&&D[tp.k]+1<dep[xt][yt][k]){//特判:要转弯即使直走得更长也没用
dep[xt][yt][k]=D[tp.k]+1;
f[xt][yt][k]=1;
q[++qr]=(Marco){xt,yt,k};
}
}
}
}
return -1;//被困住了,到不了出口
}
//&Main
int main(){
scanf("%d%d",&n,&m);
scanf("%d%d%d%d",&s.x,&s.y,&t.x,&t.y);
for(int i=1;i<=n;i++)
for(int j=1;j<=m;j++)
scanf("%d",maze[i]+j);
for(int i=1,tp;i<=10;i++)
if((tp=bfs(i))!=-1) printf("%d %d\n",i,tp);
else break;
return 0;
}
我还是太蒻了。祝大家学习愉快!
题解-[HNOI2001]遥控赛车比赛的更多相关文章
- 【洛谷 P2226】 [HNOI2001]遥控赛车比赛(最短路)
题目链接 首先拆点,把每个点拆成4个点,表示到达这个点的时候赛车的朝向. 然后考虑连边. 相邻同向并且都是可以走的点直接连边权1的边. 至于怎么转向,只需在每个点\(i\)向每个方向一直拓展直到不能走 ...
- 洛谷 题解 UVA12661 【有趣的赛车比赛 Funny Car Racing】
[题意] 在一个赛车比赛中,赛道有\(n(n<=300)\)个交叉点和\(m(m<=50000)\)条单向道路.有趣的是,每条道路都是周期性关闭的.每条道路用5个整数\(u,v,a,b,t ...
- 【水水水】【洛谷 U4566】赛车比赛
题目背景 kkk在赛车~ 题目描述 现在有N辆赛车行驶在一条直线跑道(你可以认为跑道无限长)上.它们各自以某种速度匀速前进,如果有两辆车A车和B车,A车在B车的后面,且A车的速度大于B车的速度,那么经 ...
- 赛车比赛(洛谷U4566)
题目背景 kkk在赛车~ 题目描述 现在有N辆赛车行驶在一条直线跑道(你可以认为跑道无限长)上.它们各自以某种速度匀速前进,如果有两辆车A车和B车,A车在B车的后面,且A车的速度大于B车的速度,那么经 ...
- UVA 12661 Funny Car Racing 有趣的赛车比赛(最短路,变形)
题意:赛道有n个交叉点,和m条单向路径(有重边),每条路都是周期性关闭的,且通过仍需一段时间.在比赛开始时,所有道路刚好打开,选择进入该道路必须满足“在打开的时间段进入,在关闭之前出来”,即不可在路上 ...
- 【题解】HNOI2013比赛
[题解][P3230 HNOI2013]比赛 将得分的序列化成样例给的那种表格,发现一行和一列是同时确定的.这个表格之前是正方形的,后来长宽都减去一,还是正方形.问题形式是递归的.这就启示我们可以把这 ...
- BZOJ3190[JLOI2013]赛车
Description 这里有一辆赛车比赛正在进行,赛场上一共有N辆车,分别称为个g1,g2--gn.赛道是一条无限长的直线.最初,gi位于距离起跑线前进ki的位置.比赛开始后,车辆gi将会以vi单位 ...
- BZOJ 3190 赛车 | 计算几何
BZOJ 3190 赛车 题面 这里有一辆赛车比赛正在进行,赛场上一共有N辆车,分别称为个g1,g2--gn.赛道是一条无限长的直线.最初,gi位于距离起跑线前进ki的位置.比赛开始后,车辆gi将会以 ...
- OI队测题解:
Test 17 T1: 题目大意: 喵星系有n个星球,标号为1到n,星球以及星球间的航线形成一棵树. 所有星球间的双向航线的长度都为1.小昕要在若干个星球建矿石仓库,设立每个仓库的费用为K.对于未 ...
随机推荐
- G - Green-Red Tree Gym - 102190G
题目链接:http://codeforces.com/gym/102190/attachments 题解:我们先将前5个点分别涂上红色或者绿色,使得这两棵树在5个点中都是连通,并不存在自环(建边方式不 ...
- svn http
yum install -y httpd subversion mod_dav_svn mkdir -p /var/lib/svn cd /var/lib/svn svnadmin create de ...
- 树分治(挑战p360)
poj1741 题:http://poj.org/problem?id=1741 #include<iostream> #include<algorithm> #include ...
- bwa index|amb|ann|bwt|pac|sa
-.gapcloser.fa | > t1.fa bwa index -a bwtsw -p t1 t1.fa >t1.bwa_index.log >& #$ ll #tot ...
- python之urllib模块和requests模块
一.urllib模块 python标准库自带的发送网络请求的模块. # 用python怎么打开浏览器,发送接口请求 import urllib from urllib.request import u ...
- python爬虫和数据分析、数据挖掘
一.python爬虫脑图: 二.python爬虫流程: 三.python数据分析简介 四.python数据预处理方法 五.python数据挖掘 六.数据探索基础
- signal——信号集
1.信号集 每个进程都有一个信号屏蔽字,它规定了当前要阻塞递送到该进程的信号集.对于每种可能的信号,该屏蔽字中都有一bit位与之对应.信号数可能会超过一个整型数所包含的二进制位数,因此POSIX.1 ...
- Spring Security Config : 注解 EnableWebSecurity 启用Web安全
https://blog.csdn.net/andy_zhang2007/article/details/90023901
- Qt QString的arg()方法的使用
1.QString的arg()方法用于填充字符串中的%1,%2...为给定的参数,如 QString m = tr("); // m = "12:60:60: 2.它还有另外一种重 ...
- Luogu_2279_[HNOI2003]消防局的设立
题目描述 2020年,人类在火星上建立了一个庞大的基地群,总共有n个基地.起初为了节约材料,人类只修建了n-1条道路来连接这些基地,并且每两个基地都能够通过道路到达,所以所有的基地形成了一个巨大的树状 ...