P6474 [NOI Online #2 入门组] 荆轲刺秦王 题解
荆轲将会臭名昭著
首先 $15$ 做法很简单,那就是直接 `cout<<-1`
考虑用 BFS 来解思路很简单,但是怎么求每个士兵的控制范围呢?
直接暴力时间复杂度是 $O(nma^2)$ 当然过不了一定会TLE。
所以,只需要差分+前缀和即可。
说起来简单,实现起来也简单。
然后,单打广搜大家应该都会了,可是出题人的恶魔 $18$ 点可以让你的代码慢到爆炸!
怎么剪枝呢?也很简单。
只需将当前步数大于之前已走到终点点数的点删去就可以了。
就这么简单,但我还是打了5个多小时。
然后...
正片开始!
起初,神创造了差分。标记数组是空虚混沌的,渊面黑暗,神的灵运行在差分的左右端点上。
for(int i=1;i<=n;i++)
{
for(int j=1;j<=m;j++)
{
scanf("%s",s);
if(s[0]=='S')
{
sx=i;
sy=j;
}
else if(s[0]=='T')
{
tx=i;
ty=j;
}
else if(s[0]!='.')
{
pers[i][j]=true;
int len=strlen(s),num=0;
for(int k=0;k<len;k++)
{
num=num*10+s[k]-'0';
}
for(int t=max(1,i-num+1);t<=min(n,i+num-1);t++)
{
int l=max(1,j-num+1+abs(i-t));
int r=min(m,j+num-1-abs(i-t));
cov[t][l]++;
cov[t][r+1]--;
}
}
}
}
神说差分数组要有值,与是差分数组变有了值。
for(int i=1;i<=n;i++)
{
for(int j=1;j<=m;j++)
{
cov[i][j]+=cov[i][j-1];
}
}
神看这值是对的,与是就把差分数组的预处理分为两段。有晚上,有早晨,是第一日。
for(int i=1;i<=n;i++)
{
for(int j=1;j<=m;j++)
{
scanf("%s",s);
if(s[0]=='S')
{
sx=i;
sy=j;
}
else if(s[0]=='T')
{
tx=i;
ty=j;
}
else if(s[0]!='.')
{
pers[i][j]=true;
int len=strlen(s),num=0;
for(int k=0;k<len;k++)
{
num=num*10+s[k]-'0';
}
for(int t=max(1,i-num+1);t<=min(n,i+num-1);t++)
{
int l=max(1,j-num+1+abs(i-t));
int r=min(m,j+num-1-abs(i-t));
cov[t][l]++;
cov[t][r+1]--;
}
}
}
} for(int i=1;i<=n;i++)
{
for(int j=1;j<=m;j++)
{
cov[i][j]+=cov[i][j-1];
}
}
神说数组之间要有 BFS 将其应用,神就造出了 BFS,将 BFS 以外叫做#?,BFS 以内叫做?#,这样就成了,神称 BFS 为函数,有晚上,有早晨,是第二日。
void bfs()
{
queue<node>q;
q.push(#??#);
}
...
OK,95分大成!
满分做法啊...
[好好研究](https://www.luogu.com.cn/discuss/281304)[.](https://www.luogu.com.cn/paste/fpmecy7i)
当然,上面说了,自己改吧!
我把95分的贴出来,100的就不贴了。
const int N=360,M=20;
int n,m,c1,c2,dist;
int sx,sy,tx,ty;
int cov[N][N];
bool pers[N][N];
struct node
{
int x,y,u1,u2;
node(int v1,int v2,int v3,int v4) :
x(v1),y(v2),u1(v3),u2(v4) {}
};
int d[N][N][M][M];
int vis[N][N][M][M];
const int dx[8]={-1,0,1,0,-1,1,1,-1},dy[8]={0,1,0,-1,1,1,-1,-1};
bool cmp(int x,int y)
{
return 1<=x&&x<=n&&1<=y&&y<=m;
}
void bfs()
{
queue<node>q;
d[sx][sy][0][0]=0;
vis[sx][sy][0][0]=true;
q.push((node){sx,sy,0,0});
while(!q.empty())
{
node u=q.front();
q.pop();
for(int i=0;i<8;i++)
{
int nx=u.x+dx[i];
int ny=u.y+dy[i];
if(cmp(nx,ny)&&!pers[nx][ny])
{
//cout<<nx<<" "<<ny<<" "<<u.x<<" "<<u.y<<" ";
if(cov[nx][ny])
{
if(u.u1+1<=c1&&!vis[nx][ny][u.u1+1][u.u2])
{
d[nx][ny][u.u1+1][u.u2]=d[u.x][u.y][u.u1][u.u2]+1;
vis[nx][ny][u.u1+1][u.u2]=true;
//cout<<nx<<" "<<ny<<"\n";
q.push((node){nx,ny,u.u1+1,u.u2});
}
}
else
{
if(!vis[nx][ny][u.u1][u.u2])
{
d[nx][ny][u.u1][u.u2]=d[u.x][u.y][u.u1][u.u2]+1;
vis[nx][ny][u.u1][u.u2]=true;
//cout<<nx<<" "<<ny<<"\n";
q.push((node){nx,ny,u.u1,u.u2});
}
}
}
}
if(u.u2+1<=c2)
{
for(int i=0;i<4;i++)
{
int nx=u.x+dx[i]*dist,ny=u.y+dy[i]*dist;
if(cmp(nx,ny)&&!pers[nx][ny])
{
if(cov[nx][ny])
{
if(u.u1+1<=c1&&!vis[nx][ny][u.u1+1][u.u2+1])
{
d[nx][ny][u.u1+1][u.u2+1]=d[u.x][u.y][u.u1][u.u2]+1;
vis[nx][ny][u.u1+1][u.u2+1]=true;
//cout<<nx<<" "<<ny<<"\n";
q.push((node){nx,ny,u.u1+1,u.u2+1});
}
}
else
{
if(!vis[nx][ny][u.u1][u.u2+1])
{
d[nx][ny][u.u1][u.u2+1]=d[u.x][u.y][u.u1][u.u2]+1;
vis[nx][ny][u.u1][u.u2+1]=true;
//cout<<nx<<" "<<ny<<"\n";
q.push((node){nx,ny,u.u1,u.u2+1});
}
}
}
}
}
}
return ;
}
int main()
{
cin>>n>>m>>c1>>c2>>dist;
char s[4];
for(int i=1;i<=n;i++)
{
for(int j=1;j<=m;j++)
{
scanf("%s",s);
if(s[0]=='S')
{
sx=i;
sy=j;
}
else if(s[0]=='T')
{
tx=i;
ty=j;
}
else if(s[0]!='.')
{
pers[i][j]=true;
int len=strlen(s),num=0;
for(int k=0;k<len;k++)
{
num=num*10+s[k]-'0';
}
for(int t=max(1,i-num+1);t<=min(n,i+num-1);t++)
{
int l=max(1,j-num+1+abs(i-t));
int r=min(m,j+num-1-abs(i-t));
//cout<<l<<" "<<r<<"\n";
cov[t][l]++;
cov[t][r+1]--;
}
}
}
}
//cout<<"qidian "<<sx<<" "<<sy<<" zhongdian "<<tx<<" "<<ty<<"\n";
for(int i=1;i<=n;i++)
{
for(int j=1;j<=m;j++)
{
cov[i][j]+=cov[i][j-1];
}
}
/*for(int i=1;i<=n;i++)
{
for(int j=1;j<=m;j++)
{
cout<<cov[i][j]<<" ";
}
cout<<"\n";
}*/
bfs();
int t=n*m,u1,u2;
for(int i=0;i<=max(c1,c2);i++)
{
for(int j=0;j<=max(c1,c2);j++)
{
if(vis[tx][ty][j][i]&&d[tx][ty][j][i]<t)
{
t=d[tx][ty][j][i];
u1=j;
u2=i;
} }
}
if(t==n*m)
{
cout<<-1;
}
else
{
cout<<t<<" "<<u1<<" "<<u2;
}
return 0;
}
P6474 [NOI Online #2 入门组] 荆轲刺秦王 题解的更多相关文章
- P6474 [NOI Online #2 入门组] 荆轲刺秦王
P6474 [NOI Online #2 入门组] 荆轲刺秦王 bfs+差分+卡常 本来我其实是场内选手,但是因为记错提交时间,晚了半小时才交,交不上了,就自动降级为了场外选手 题面复杂,不简述了 首 ...
- P7473 [NOI Online 2021 入门组] 重力球
P7473 [NOI Online 2021 入门组] 重力球 题意 给你一个正方形平面,某些位置有障碍,对于平面上两个球,每次你可以改变重力方向使两个球下落到最底端,求使两个球位置重合的最小改变重力 ...
- 洛谷 P6189 - [NOI Online #1 入门组]跑步(根号分治+背包)
题面传送门 题意: 求有多少个数列 \(x\) 满足: \(\sum x_i=n\) \(x_i\geq x_{i+1}\) 答案对 \(p\) 取模. ...你确定这叫"入门"组 ...
- NOI Online 2021 入门组 T1
Description 题目描述 Alice.Bob 和 Cindy 三个好朋友得到了一个圆形蛋糕,他们打算分享这个蛋糕. 三个人的需求量分别为 \(a, b, c\),现在请你帮他们切蛋糕,规则如下 ...
- [NOI 2020 Online] 入门组T1 文具采购(洛谷 P6188)题解
原题传送门 题目部分:(来自于考试题面,经整理) [题目描述] 小明的班上共有 n 元班费,同学们准备使用班费集体购买 3 种物品: 1.圆规,每个 7 元. 2.笔,每支 4 元. 3.笔记本,每本 ...
- NOI Online #1 入门组 魔法
全网都是矩阵快速幂,我只会倍增DP 其实这题与 AcWing 345. 牛站 还是比较像的,那题可以矩阵快速幂 / 倍增,这题也行. 先 \(Floyd\) 预处理两点之间不用魔法最短距离 \(d_{ ...
- [题解] [NOI Online 2021 入门组 T3] 重力球
题目大意 在一个 \(n\times n\) 的矩形中,题目会给出 \(m\) 个障碍物.有两个小球,你可以选定四个方向(上下左右)的其中一个,小球会朝着这四个方向一直滚动,直到遇到障碍物或是矩形的边 ...
- P6189 [NOI Online #1 入门组] 跑步 (DP/根号分治)
(才了解到根号分治这样的妙方法......) 将每个数当成一种物品,最终要凑成n,这就是一个完全背包问题,复杂度O(n2),可以得80分(在考场上貌似足够了......) 1 #include < ...
- 【NOI Online 2020】入门组 总结&&反思
前言: 这次的NOI Online 2020 入门组我真的无力吐槽CCF的网站了,放段自己写的diss的文章,供一乐 如下:(考试后当天晚上有感而发) 今天是个好日子!!!(我都经历了什么...... ...
- NOI ONLINE 入门组 魔法 矩阵快速幂
做了这道题我才发现NOI入门组!=NOIP普及组 题目链接 https://www.luogu.com.cn/problem/P6190 题意 给出一张有向图,你有K次机会可以反转一条边的边权,即让它 ...
随机推荐
- [29] CSP模拟2
A.不相邻集合 考虑值域上连续的段,可以发现连续的长度为 \(x\) 的段的贡献必定为 \(\lceil{\frac{x}{2}}\rceil\) 考虑并查集维护值域连续的段的大小,每次询问求出全部连 ...
- Brainstorm 了道题但是不会做
题 因为没想出来暂时没定数据范围,不过应该会在 \(n^{2}\) 到 \(n^{3}\) 级别 我的一个思路是先对合法的方案连并查集,然后并查集内判重,但是不会算方案数,因为假如找到重的了不能直接看 ...
- Windows安装Powershell7.x
事件起因: 由于需要运行一个脚本,但是该脚本是广大网友群众使用Powershell7写的,我自带的是Powershell5,运行过程中总是出现莫名其妙的问题,于是决定将Powershell升级到Pow ...
- Java使用snmp协议采集服务器信息
Java使用snmp协议采集服务器信息 一.snmp简介 SNMP是专门设计用于在 IP 网络管理网络节点(服务器.工作站.路由器.交换机及Hubs等)的一种标准协议,它是一种应用层协议. ...
- DWC3和XHCI的区别
DWC3(DesignWare USB 3.0 Controller)和XHCI(eXtensible Host Controller Interface)都是与USB控制器相关的技术,但它们的作用和 ...
- USB通讯架构及数据模型
注意: (1)一个usb设备由一个或者多个接口组成: (2)每一个接口为usb设备的一个功能,比如上面的usb设备由两个接口,一个可用于鼠标,一个可用于键盘: (3)每个接口占用usb设备的多个端口资 ...
- iOS工厂模式使用小结
一.什么是工厂方法? 正式的解释是:在基类中定义创建对象的一个接口,让子类决定实例化哪个类.工厂方法让一个类的实例化延迟到子类中进行.工厂方法要解决的问题是对象的创建时机,它提供了一种扩展的策略,很好 ...
- computed 中的属性名和data的属性名可以相同吗?
不可以,因为无论computed 中的属性名还是 data 又或者是props中的属性名,都会别挂载到组件实例上,所以名字都不允许重复 : ps:好比一个作用域里面不允许定义2个相同的变量名 :
- C#多线程编程:AutoResetEvent
作用 简单的完成多线程同步,两个线程共享相同的AutoResetEvent对象.线程可以通过调用AutoResetEvent对象的WaitOne()方法进入等待状态当第二个线程调用Set()方法时,它 ...
- 网页设计中常用的Web英文安全字体
原文地址:https://www.openkee.com/post-176.html 在 Web 编码中,CSS 默认应用的 Web 字体是有限的,你能看到的字体别人未必看得到.虽然在新版本的CSS3 ...