【图论经典题目讲解】洛谷 P5304 旅行者

这篇具有很好参考价值的文章主要介绍了【图论经典题目讲解】洛谷 P5304 旅行者。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

P5304 旅行者

D e s c r i p t i o n \mathrm{Description} Description

给定一个 n n n 个点, m m m 条边的有向图,求解 k k k 个点两两间最短路长度的最小值。

S o l u t i o n \mathrm{Solution} Solution

对于 k k k 个点,可以考虑二进制分组优化,即对于每一位为 1 1 1 的点放入 1 1 1 组(设为 A A A 组),为 0 0 0 的点放入 1 1 1 组(设为 B B B 组)。

则如果建立一个虚拟源点和一个虚拟汇点,并由虚拟源点向 A A A 组中的每一个点连一条长度为 0 0 0 的边,且由 B B B 组中的每一个点向虚拟汇点连一条长度为 0 0 0 的边,那么虚拟源点到虚拟汇点的最短路即为 A A A 组中任意一个点到 B B B 组中任意一个点的最短路长度的最小值。

正确性证明:对于 k k k 个点中的任意 2 2 2 个点,由于编号是不同的,那么必定有一位二进制是不同的,所以总有一次一个点在 A A A 组,一个点在 B B B 组,这 2 2 2 个点之间的最短路也一定会给答案贡献 1 1 1 次。文章来源地址https://www.toymoban.com/news/detail-826662.html

C o d e Code Code

#include <bits/stdc++.h>
#define int long long

using namespace std;

typedef pair<int, int> PII;
typedef long long LL;

const int SIZE = 4e7 + 10, SIZE2 = 1e5 + 10;

int N, M, K;
int h[SIZE2], hs[SIZE2], e[SIZE], ne[SIZE], w[SIZE], idx;
int A[SIZE2], Dist[SIZE2], Vis[SIZE2];

void add(int h[], int a, int b, int c)
{
	e[idx] = b, ne[idx] = h[a], w[idx] = c, h[a] = idx ++;
}

int Dijkstra(int S, int T)
{
	memset(Dist, 0x3f, sizeof Dist);
	memset(Vis, 0, sizeof Vis);
	priority_queue<PII, vector<PII>, greater<PII>> Heap;

	Heap.push({0, S}), Dist[S] = 0;
	while (Heap.size())
	{
		auto Tmp = Heap.top();
		Heap.pop();

		int u = Tmp.second;
		if (Vis[u]) continue;
		Vis[u] = 1;

		for (int i = hs[u]; ~i; i = ne[i])
		{
			int j = e[i];
			if (Dist[j] > Dist[u] + w[i])
			{
				Dist[j] = Dist[u] + w[i];
				Heap.push({Dist[j], j});
			}
		}
	}

	return Dist[T];
}

void solve()
{
	memset(h, -1, sizeof h);
	memset(hs, -1, sizeof hs);

	cin >> N >> M >> K;

	int u, v, c;
	while (M --)
	{
		cin >> u >> v >> c;
		add(h, u, v, c);
	}

	for (int i = 1; i <= K; i ++)
		cin >> A[i];

	memcpy(hs, h, sizeof h);

	int S = 0, T = N + 1, Result = 1e18;
	for (int i = 0; i < 17; i ++)
	{
		for (int j = 1; j <= K; j ++)
			if (A[j] >> i & 1)
				add(hs, S, A[j], 0);
			else
				add(hs, A[j], T, 0);
		Result = min(Result, Dijkstra(S, T));
		memcpy(hs, h, sizeof h);
		for (int j = 1; j <= K; j ++)
			if (A[j] >> i & 1)
				add(hs, A[j], T, 0);
			else
				add(hs, S, A[j], 0);
		Result = min(Result, Dijkstra(S, T));
		memcpy(hs, h, sizeof h);
	}

	cout << Result << endl;
}

signed main()
{
	cin.tie(0);
	cout.tie(0);
	ios::sync_with_stdio(0);

	int Data;

	cin >> Data;

	while (Data --)
		solve();

	return 0;
}

到了这里,关于【图论经典题目讲解】洛谷 P5304 旅行者的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包赞助服务器费用

相关文章

  • 【SQL】1407. 排名靠前的旅行者

    【SQL】1407. 排名靠前的旅行者

    leetcode题目:1407. 排名靠前的旅行者 先过滤,再连表 先连表,再过滤

    2024年03月22日
    浏览(10)
  • 「SQL面试题库」 No_105 排名靠前的旅行者

    「SQL面试题库」是由 不是西红柿 发起,全员免费参与的SQL学习活动。我每天发布1道SQL面试真题,从简单到困难,涵盖所有SQL知识点,我敢保证只要做完这100道题,不仅能轻松搞定面试,代码能力和工作效率也会有明显提升。 1.1 活动流程 整理题目 :西红柿每天无论刮风下雨

    2024年02月16日
    浏览(7)
  • 简单图论:旅行

    最小生成树(Kruskal算法) 不同的的最小生成树: 不是连通的权重最小; 而是连通起始点和终点的路径上最大最小速度比值最小。 如何使得速度比值最小: 问题1:什么情况下比值最小? ( 1 、 3 、 5 、 4 、 2 ) ⇒ ( 1 、 2 、 3 、 4 、 5 ) (1、3、5、4、2) Rightarrow (1、2、

    2024年02月11日
    浏览(4)
  • 旅行商问题(TSP)及Python图论求解

    旅行商问题(Traveling Salesman Problem, TSP)是指给定一个城市的集合以及每两个城市之间的距离,找到一条经过每个城市恰好一次且路径最短的回路。 可以想象成一个旅行商要拜访多个城市,问他如何安排路线,使得行程最短。这个问题可能看起来简单,但是随着城市数量的增

    2024年02月12日
    浏览(6)
  • 【深度优先搜索】【图论】【树】2646. 最小化旅行的价格总和

    【深度优先搜索】【图论】【树】2646. 最小化旅行的价格总和

    【数位dp】【动态规划】【状态压缩】【推荐】1012. 至少有 1 位重复的数字 深度优先搜索 图论 树 现有一棵无向、无根的树,树中有 n 个节点,按从 0 到 n - 1 编号。给你一个整数 n 和一个长度为 n - 1 的二维整数数组 edges ,其中 edges[i] = [ai, bi] 表示树中节点 ai 和 bi 之间存在

    2024年02月19日
    浏览(15)
  • 洛谷题单 -- 图论的简单入门

    洛谷题单 -- 图论的简单入门

    图的存储 - 洛谷 这一题要考察图的存储方式 , 一般可以使用邻接矩阵 或 邻接表来存储 图的结点 和1 边的信息 ,详情请看代码 :  【深基18.例3】查找文献 - 洛谷 这题考察有向图的 dfs 和 bfs ,详情请看代码,如果用邻接矩阵的话一定会mle,只能够使用邻接表,我这里采用的是用

    2024年04月13日
    浏览(6)
  • 【算法每日一练]-图论(保姆级教程篇15 )#会议(模板题) #医院设置 #虫洞(模板题) #无序字母对 #旅行计划 #最优贸易

    【算法每日一练]-图论(保姆级教程篇15 )#会议(模板题) #医院设置 #虫洞(模板题) #无序字母对 #旅行计划 #最优贸易

    目录 今日知识点: 求数的重心先dfs出d[1]和cnt[i],然后从1进行dp求解所有d[i] 两两点配对的建图方式,检查是否有环 无向图欧拉路径+路径输出 topo+dp求以i为终点的游览城市数 建立分层图转化盈利问题成求最长路 会议(模板题) 医院设置  虫洞(模版题) 无序字母对  旅行

    2024年01月21日
    浏览(9)
  • 【算法每日一练]-图论(保姆级教程篇14 )#会议(模板题) #医院设置 #虫洞(模板题) #无序字母对 #旅行计划 #最优贸易

    【算法每日一练]-图论(保姆级教程篇14 )#会议(模板题) #医院设置 #虫洞(模板题) #无序字母对 #旅行计划 #最优贸易

    目录 今日知识点: 求数的重心先dfs出d[1]和cnt[i],然后从1进行dp求解所有d[i] 两两点配对的建图方式,检查是否有环 无向图欧拉路径+路径输出 topo+dp求以i为终点的游览城市数 建立分层图转化盈利问题成求最长路 会议(模板题) 医院设置  虫洞(模版题) 无序字母对  旅行

    2024年02月02日
    浏览(11)
  • 算法题目题单+题解——图论

    本文为自己做的一部分图论题目,作为题单列出,持续更新。 题单由题目链接和题解两部分组成,题解部分提供简洁题意,代码仓库:Kaiser-Yang/OJProblems。 对于同一个一级标题下的题目,题目难度尽可能做到递增。 题目链接:Luogu P3547 [POI2013] CEN-Price List 题解: 题目链接:

    2024年02月19日
    浏览(9)
  • PAT甲级图论相关题目

    PAT甲级图论相关题目

    PAT甲级图论相关题目: 分数 25 As an emergency rescue team leader of a city, you are given a special map of your country. The map shows several scattered cities connected by some roads. Amount of rescue teams in each city and the length of each road between any pair of cities are marked on the map. When there is an emergency call to you from some o

    2024年01月21日
    浏览(8)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包