PTA拯救007(升级版)

在老电影“007之生死关头”(Live and Let Die)中有一个情节,007被毒贩抓到一个鳄鱼池中心的小岛上,他用了一种极为大胆的方法逃脱 —— 直接踩着池子里一系列鳄鱼的大脑袋跳上岸去!(据说当年替身演员被最后一条鳄鱼咬住了脚,幸好穿的是特别加厚的靴子才逃过一劫。)

设鳄鱼池是长宽为100米的方形,中心坐标为 (0, 0),且东北角坐标为 (50, 50)。池心岛是以 (0, 0) 为圆心、直径15米的圆。给定池中分布的鳄鱼的坐标、以及007一次能跳跃的最大距离,你需要给他指一条最短的逃生路径 —— 所谓“最短”是指007要跳跃的步数最少。

输入格式:

首先第一行给出两个正整数:鳄鱼数量 N(≤100)和007一次能跳跃的最大距离 D。随后 N 行,每行给出一条鳄鱼的 (x,y) 坐标。注意:不会有两条鳄鱼待在同一个点上。

输出格式:

如果007有可能逃脱,首先在第一行输出007需要跳跃的最少步数,然后从第二行起,每行给出从池心岛到岸边每一步要跳到的鳄鱼的坐标 (x,y)。如果没可能逃脱,就在第一行输出 0 作为跳跃步数。如果最短路径不唯一,则输出第一跳最近的那个解,题目保证这样的解是唯一的。

输入样例 1:

17 15
10 -21
10 21
-40 10
30 -50
20 40
35 10
0 -10
-25 22
40 -40
-30 30
-10 22
0 11
25 21
25 10
10 10
10 35
-30 10

输出样例 1:

4
0 11
10 21
10 35

输入样例 2:

4 13
-12 12
12 12
-12 -12
12 -12

输出样例 2:

0

思路:

一个一个尝试能不能跳,看最后能不能跳出去,记录路径。

方法:

用bfs搜索一下看能不能跳出去,顺便记录路径。

bfs代码:

void bfs(){
	bool flag=1;//当前是否可以跳出边界 
	ll k=q.size();B++;//当前可以跳到的鳄鱼数量,跳跃步数+1 
	for(ll i = 0 ; i < k ; i ++){
		ll x=q.front().first.first ,y=q.front().first.second ,z=q.front().second;
		q.pop();
		if(x+d >= 50 || y+d >= 50 || x-d <= -50 || y-d <= -50)//当前节点可以跳出 
			js.push(z),flag=0;//记录节点 
		ll len=e.size();//尝试之前还跳不到的鳄鱼 
		for(ll j = 0 ; j < len ; j ++){
			ll a=e.front().first.first ,b=e.front().first.second ,c=e.front().second;
			e.pop();
			if(d*d >= (a-x)*(a-x)+(b-y)*(b-y))q.push({{a,b},c}) ,v[c].w = z;//存当前c可以从z跳过来 
			else e.push({{a,b},c});//把鳄鱼放回去 
		}
	}
	if(flag && !q.empty())bfs();//看是否还要继续跳 
}

AC代码:

#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
#define endl "\n"
#define P pair<ll,ll>

const ll N = 1e2+7;
ll n,B=1;//n只鳄鱼,记录步数 
double d;//跳跃的距离 
struct node{
	ll x,y,w;//鳄鱼在x,y点,可以从w点跳到当前点 
}v[N];
queue<ll>js;//哪些节点在最少步数的情况可以跳出去 
queue<pair<P,ll> >q,e;//当前可以跳到的鳄鱼,还跳不到的鳄鱼 
stack<P>st;//反向输出存一下栈 

void bfs(){
	bool flag=1;//当前是否可以跳出边界 
	ll k=q.size();B++;//当前可以跳到的鳄鱼数量,跳跃步数+1 
	for(ll i = 0 ; i < k ; i ++){
		ll x=q.front().first.first ,y=q.front().first.second ,z=q.front().second;
		q.pop();
		if(x+d >= 50 || y+d >= 50 || x-d <= -50 || y-d <= -50)//当前节点可以跳出 
			js.push(z),flag=0;//记录节点 
		ll len=e.size();//尝试之前还跳不到的鳄鱼 
		for(ll j = 0 ; j < len ; j ++){
			ll a=e.front().first.first ,b=e.front().first.second ,c=e.front().second;
			e.pop();
			if(d*d >= (a-x)*(a-x)+(b-y)*(b-y))q.push({{a,b},c}) ,v[c].w = z;//存当前c可以从z跳过来 
			else e.push({{a,b},c});//把鳄鱼放回去 
		}
	}
	if(flag && !q.empty())bfs();//看是否还要继续跳 
}

void solve(){
	cin >> n >> d;
	if(d >= 50-7.5){//特判一步直接出逃的方案 
		cout << B << endl;
		return;
	}
	ll x,y;
	for(ll i = 0 ; i < n ; i ++){
		cin >> x >> y;
		v[i].x=x,v[i].y=y,v[i].w=-1;//初始没有任何点可以跳到这个点 
		(d+7.5)*(d+7.5) >= x*x+y*y ? q.push({{x,y},i}) : e.push({{x,y},i});//判哪些点可以一步跳到 
	}
	bfs();
	if(js.empty())cout << 0 << endl;//无法逃出 
	else{
		ll jl=1e9,w=-1;
		cout << B << endl;
		while(!js.empty()){//查找每个最少步数出逃点第一步跳到谁 
			x=y=js.front();
			js.pop();
			while(v[x].w != -1)x=v[x].w;
			if(v[x].x*v[x].x+v[x].y*v[x].y < jl)//比较第一步最短的是谁 
				jl=v[x].x*v[x].x+v[x].y*v[x].y,w=y;
		}
		while(v[w].w != -1)//路径回溯 
			st.push({v[w].x,v[w].y}),w=v[w].w;
		cout << v[w].x << " " << v[w].y << endl;
		while(!st.empty())//打印路径 
			cout << st.top().first << " " << st.top().second << endl,
			st.pop();
	}
	return;
}

int main(){
	ll t=1;//cin >> t;
	while(t --)solve();
	return 0;
}

  • 2
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值