首页 > 其他分享 >拯救大兵瑞恩 孤岛营救

拯救大兵瑞恩 孤岛营救

时间:2024-08-02 15:06:02浏览次数:11  
标签:calcpostate int st 瑞恩 孤岛 100 营救 单元

// 拯救大兵瑞恩.cpp : 此文件包含 "main" 函数。程序执行将在此处开始并结束。
//



/*
* http://ybt.ssoier.cn:8088/problem_show.php?pid=1495
* https://loj.ac/p/6121
* https://www.luogu.com.cn/problem/P4011
【题目描述】
1944 年,特种兵麦克接到国防部的命令,要求立即赶赴太平洋上的一个孤岛,营救被敌军俘虏的大兵瑞恩。瑞恩被关押在一个迷宫里,
迷宫地形复杂,但幸好麦克得到了迷宫的地形图。迷宫的外形是一个长方形,其南北方向被划分为 n 行,东西方向被划分为 m 列, 
于是整个迷宫被划分为 n×m 个单元。每一个单元的位置可用一个有序数对 (单元的行号, 单元的列号) 来表示。
南北或东西方向相邻的 2 个单元之间可能互通,也可能有一扇锁着的门,或者是一堵不可逾越的墙。
迷宫中有一些单元存放着钥匙,并且所有的门被分成 p 类, 打开同一类的门的钥匙相同,不同类门的钥匙不同。

大兵瑞恩被关押在迷宫的东南角,即 (n,m) 单元里,并已经昏迷。迷宫只有一个入口, 在西北角。
也就是说,麦克可以直接进入 (1,1) 单元。另外,麦克从一个单元移动到另一个相邻单元的时间为 1,
拿取所在单元的钥匙的时间以及用钥匙开门的时间可忽略不计。

试设计一个算法,帮助麦克以最快的方式到达瑞恩所在单元,营救大兵瑞恩。

【输入】
第一行有三个整数,分别表示 n,m,p 的值。

第二行是一个整数k,表示迷宫中门和墙的总数。

第 i+2 行 (1≤i≤k),有 5 个整数,依次为 xi1,yi1,xi2,yi2,gi:当 gi≥1 时,表示 (xi1,yi1)单元与 (xi2,yi2) 单元之间有一扇第 gi 类的门,
当 gi=0 时, 表示 (xi1,yi1) 单元与 (xi2,yi2) 单元之间有一堵不可逾越的墙。

第 k+3 行是一个整数 s,表示迷宫中存放的钥匙总数。

第 k+3+j 行 (1≤j≤s) ,有 3 个整数,依次为 xi1,yi1,qi​​​​ ,表示第 j 把钥匙存放在 (xi1,yi1) 单元里,并且第 j 把钥匙是用来开启第 qi 类门。

输入数据中同一行各相邻整数之间用一个空格分隔。

【输出】
输出麦克营救到大兵瑞恩的最短时间。如果问题无解,则输出 −1。

【输入样例】
4 4 9
9
1 2 1 3 2
1 2 2 2 0
2 1 2 2 0
2 1 3 1 0
2 3 3 3 0
2 4 3 4 1
3 2 3 3 0
3 3 4 3 0
4 3 4 4 0
2
2 1 2
4 2 1
【输出样例】
14
【提示】
|xi1−xi2|+|yi1−yi2|=1,0≤gi≤p
1≤qi≤p
n,m,p≤10,k<150
*/

#include <iostream>
#include <map>
#include <queue>
#include <unordered_map>

using namespace std;


int n, m,p;
int k, s;

int calcid(int x, int y) {
	return x + y * 100;
}
void getposFromid(int n, int& x, int& y) {
	x = n % 100;
	y = n / 100;
}

int callongid(int x, int y, int a, int b) {
	return x + y * 100 + a * 10000 + b * 1000000;
}

void getposFromlongid(int n, int& x, int& y,int &a,int &b) {
	x = n % 100;
	y = n%10000 / 100;
	a = n % 1000000 / 10000;
	b = n / 1000000;
}

int calcpostate(int x, int y, int st) {
	return x + y * 100 + st * 10000;
}

void getposstatefromNum(int n, int& x, int& y, int& st) {
	x = n % 100;
	y = n % 10000 / 100;
	st = n / 10000;
}


unordered_map<int, int> doors;
unordered_map<int, int> keys;
unordered_map<int, int> vis;

int addx[] = { 1,0,-1,0 };
int addy[] = { 0,1,0,-1 };


void solve() {
	queue<int> q;
	q.push(calcpostate(1,1,0));
	//如果当前位置有钥匙 拿起钥匙
	vis[calcpostate(1, 1, 0| keys[calcid(1, 1)])]  = 1;

	while (q.size()) {
		auto t = q.front(); q.pop();
		int x = 0; int y = 0; int st = 0; int step = vis[t];
		getposstatefromNum(t, x, y, st);

		if (x == 1 && y == 2 && st == 4) {
			int ds = 9;
		}

		if (x == n && y == m) {
			cout << vis[t] - 1 << endl;
			return;
		}

		for (int i = 0; i < 4; i++) {
			int a = x + addx[i]; int b = y+addy[i];
			if (a <= 0 || b <= 0 || a > n || b > m) continue;
			if (vis.count(calcpostate(a, b,st)) != 0) continue;
			//查看 xy ab之间是否有墙门
			if(doors.count(callongid(x, y, a, b)) != 0){
				int type = doors[callongid(x, y, a, b)];
				//有墙 就不必考虑
				if (type == 0) { continue; }
				else if (  (type & st)  != 0) {
					if (a == 1 && b == 3 && st == 0) {
						int ds = 9;
					}
					int newst = st | keys[calcid(a, b)];
					q.push(calcpostate(a, b, newst));
					vis[calcpostate(a, b, newst)] = step+1;
				}
			}
			else {
				if (a == 1 && b == 3 && st == 0) {
					int ds = 9;
				}
				//两个之间没有墙
				int newst = st | keys[calcid(a, b)];
				q.push(calcpostate(a, b, newst));
				vis[calcpostate(a, b, newst)] = step + 1;
			}
		}
	}
	

	cout << -1 << endl;
	return;
}

int main()
{
	cin >> n >> m >> p;
	cin >> k;
	for (int i = 1; i <= k; i++) {
		int x, y, a, b, t; cin >> x >> y >> a >> b >> t;
		if (t != 0) {
			t = 1 << t;
		}
		doors[callongid( x,  y, a, b)] = t;
		doors[callongid(a, b,x,y)] = t;
	}
	cin >> s;
	for (int i = 1; i <= s; i++) {
		int x, y, t;
		cin >> x >> y >> t;
		keys[calcid(x, y)] |= 1 << t;
	}
	solve();

	return 0;
}

标签:calcpostate,int,st,瑞恩,孤岛,100,营救,单元
From: https://www.cnblogs.com/itdef/p/18338793

相关文章

  • 大型企业如何整合集成全域数据、解决数据孤岛难题?
    今天,我们说一下大型企业全域数据的整合集成问题。通常,中大型企业和集团公司拥有大量多源异构的数据存储资源,如数据仓库、数据湖以及分布于分子公司和混合多云平台的业务系统,通过传统物理集中统一数据资产管理的方式难度高,代价大,这就带来一系列问题:数据孤岛普遍存在,包括随着业......
  • 《孤岛惊魂6》风灵月影修改器:全方位使用指南与技巧分享
    对于《孤岛惊魂6》的玩家来说,风灵月影修改器无疑是一个强大的辅助工具,它能极大地改变游戏体验,提供从无限资源到增强角色能力的各种便利。本文将详细介绍风灵月影修改器的使用方法,并分享一些高级技巧,帮助玩家掌握这一工具,享受更自由、更丰富的游戏乐趣。修改器概述风灵月影修......
  • 基于事件触发机制的孤岛微电网二次电压与频率协同控制仿真模型(Simulink仿真实现)
    ......
  • 打造企业Wiki,打通信息孤岛
    引言:在现代企业中,信息孤岛是一个普遍存在的问题,不同部门之间、不同团队之间甚至是同一团队内部都存在信息闭塞的情况。而企业Wiki作为一种强大的知识管理办法,可以帮助企业打通信息孤岛,促进信息共享和团队协作。本文将介绍如何打造企业Wiki系统,让信息流畅传递,打破沟通壁垒,提升......
  • Matlab|【核心复现】同时考虑考虑孤岛与重构的配电网故障恢复运行策略
    目录 主要内容     基本知识   1.问题引出2.可控负荷3.网络拓扑约束4.算法流程  结果一览   1.原文结果2.程序运行结果下载链接 主要内容   该模型复现文章《同时考虑考虑孤岛与重构的配电网故障恢复运行策略》,以IEEE33配电网为分析对象,通过......
  • Matlab|孤岛划分|弹性配网故障划分模型
    目录1 主要内容1.1 DistFlow模型1.2 虚拟潮流1.3 目标函数2 部分代码3 程序结果4下载链接1 主要内容程序主要复现《ANewModelforResilient DistributionSystemsbyMicrogridsFormation》,建立灾害情况下配网优化孤岛划分方案,通过虚拟潮流的方式优......
  • P4542 [ZJOI2011] 营救皮卡丘
    P4542[ZJOI2011]营救皮卡丘注意到什么叫两面包夹芝士这个是最优解这个是最劣解这究竟是怎么一回事呢?请看下文挺有趣的这道题,我们先来分析一下限制最基础的就是每个点都需要经过这一点,并且要求总路程最小很容易想到的就是路径覆盖问题,进而可以尝试费用......
  • 什么是数据孤岛?为什么对企业影响巨大?
    数据孤岛指的是数据在组织内部无法自由流通和共享的状态,这种现象不仅影响了业务的高效运作,也威胁着企业的创新和竞争力。本文将深入探讨数据孤岛问题,分析其产生的原因以及对企业的影响,最后提出有效的应对策略。一、数据孤岛的定义与原因数据孤岛并非一夜之间形成,它是由多种因素......
  • 什么是数据孤岛?哪些行业比较容易产生?
    容易产生数据孤岛的行业有:一、医疗行业;二、金融行业;三、制造业;四、零售业;五、教育行业;六、能源行业。这些行业内部存在的数据孤岛问题,不仅影响了信息的共享和整合,也妨碍了业务流程的优化和创新。一、医疗行业医疗行业是一个信息密集、分散且高度专业化的领域。不同医院、诊所和......
  • 如何打破数据孤岛,实现数据治理
      在如今大数据时代,企业或组织往往面临一个重要问题,即如何有效地治理数据孤岛。数据孤岛是指数据在企业内部或不同部门之间形成一种隔离状态,导致数据无法流通、共享和利用。在数聚看来,要解决这一问题需要经过一系列有序的步骤和措施。 首先,建立全面的数据治理意识。数据治理......