首页 > 其他分享 >题解:AT_abc369_e [ABC369E] Sightseeing Tour

题解:AT_abc369_e [ABC369E] Sightseeing Tour

时间:2024-12-06 22:12:00浏览次数:5  
标签:int 题解 短路 Tour vis abc369 810 include dis

题目大意

给定一个 $N$ 个点,$M$ 条边的无向图。其中边有边权。有 $Q$ 次询问,每一次给你 $K$ 条必须经过的边(但是方向没有限制),问从 $1$ 到 $N$ 的最短路长度是多少。

思路

观察数据范围,可以发现:虽然 $M$ 很大,但是 $N$ 和 $K$ 并不大。$K\le5$,可以暴力枚举每一条边经过时的方向以及经过它们的顺序。$N\le400$,考虑预处理出任意两点之间的最短路。

我们使用dfs来暴力枚举,首先考虑每条边的方向,再全排列去确定遍历顺序。在每一种具体方案中,我们可以将点之间的最短路相加,并且加上边权,得出这种方案的最短路径长度,再取最小值即可。

代码

#include <cstdio>
#include <iostream>
#include <algorithm>
#include <queue>
#include <vector>
#include <cstring>
using namespace std;
typedef long long LL;

int n, m, q, k;
int a[40], b[40];
LL ans; // 答案
int A[400010]; // 边
int B[400010]; // 边
int C[400010]; // 边
LL dis[810][810]; // 最短路
int vis[810][810]; // 标记

struct edge // 边
{
	int y, w;
} ;

vector<edge> g[810]; // 图

void add(int x, int y, int w) // 建边
{
	g[x].push_back((edge){y, w});
}

struct node // 点
{
	int x; // 编号
	LL d; // 距离起点的最短路
	
	bool operator < (const node & b) const // 重载运算符
	{
		return d > b.d;
	}
} ;

void dijkstra(int s) // 最短路
{
	priority_queue<node> q;
	memset(dis[s], 0x3f, sizeof(dis[s]));
	memset(vis[s], 0, sizeof(vis[s]));
	q.push((node){s, 0});
	dis[s][s] = 0;
	while (q.size())
	{
		int x = q.top().x;
		q.pop();
		if (vis[s][x])
			continue;
		vis[s][x] = 1;
		for (int i = 0; i < g[x].size(); i++)
		{
			int y = g[x][i].y;
			int w = g[x][i].w;
			if (dis[s][y] > dis[s][x] + w)
			{
				dis[s][y] = dis[s][x] + w;
				q.push((node){y, dis[s][y]});
			}
		}
	}
}

void solve(int step) // 暴力枚举
{
	if (step > k)
	{
		for (int i = 1; i <= k; i++)
			b[i] = i;
		do
		{
			LL sum = dis[1][A[a[b[1]]]];
			for (int i = 2; i <= k; i++)
				sum += C[a[b[i - 1]]] + dis[B[a[b[i - 1]]]][A[a[b[i]]]];
			sum += C[a[b[k]]] + dis[B[a[b[k]]]][n];
			ans = min(ans, sum);
		} while (next_permutation(b + 1, b + k + 1)); // 全排列
		return ;
	}
	swap(A[a[step]], B[a[step]]); // 边的方向
	solve(step + 1);
	swap(A[a[step]], B[a[step]]); // 边的方向
	solve(step + 1);
}

int main()
{
	cin >> n >> m;
	for (int i = 1; i <= m; i++)
	{
		int x, y, w;
		cin >> x >> y >> w;
		add(x, y, w);
		add(y, x, w);
		A[i] = x;
		B[i] = y;
		C[i] = w;
	}
	for (int i = 1; i <= n; i++)
		dijkstra(i);
	cin >> q;
	while (q--)
	{
		cin >> k;
		for (int i = 1; i <= k; i++)
			cin >> a[i];
		ans = 9e18; // 不要忘记初始值
		solve(1);
		cout << ans << endl;
	}
	return 0;
} 

标签:int,题解,短路,Tour,vis,abc369,810,include,dis
From: https://www.cnblogs.com/thrift/p/18591510

相关文章

  • 题解:CF1950F 0, 1, 2, Tree!
    题目链接思路不能形成树的情况:第一,一棵树必须有叶子节点。所以$c=0$的情况就一定不能形成一棵树。其次,可以发现,我们每增加一个度为$2$的节点,叶子节点就也会增加$1$个。所以$a+1\neqc$的情况也肯定不行了。代码片段if(!c||a+1!=c) cout<<"-1"<<endl;......
  • 题解:P4009 汽车加油行驶问题
    题目思路这是一个分层图最短路问题,我们可以使用升维的方法来完成本题。因为存在加油付费的问题,边权不一定为$1$,所以不能使用广搜来做。数据范围不大:$N\le100$。可以使用SPFA算法完成本题。每一个状态有三个值,分别是当前到达的行、列,以及剩下的油还能走几步。考虑是否需要加油......
  • 题解:P2217 [HAOI2007] 分割矩阵
    思路首先,我们要弄明白题中的方差是什么。公式:$S=\sqrt{\frac{1}{n}\sum_{i=1}^{n}(x_i-\bar{x})^2}$接下来,我们思考一下题目怎么做。数据很小,于是想到了暴搜。但是时间复杂度有点难以接受啊,优化一下吧。有一种很有效的优化,那就是广为人知的记忆化搜索。它能使所有......
  • 题解:AT_abc368_d[ABC368D] Minimum Steiner Tree
    题目大意题目给定一棵由$N$个节点组成的无根树,删除其中的一些点和边,使剩下的点和边仍然能够组成一棵树,且包含给定的$K$个特殊点,问最少剩下几个点。思路我们可以发现,这棵无根树的根必须是给定的特殊点之一,不然根节点就可以删除,答案就不是最优。所以我们使用深度优先搜索遍......
  • 题解:AT_abc368_c [ABC368C] Triple Attack
    思路$N$很大,导致$T$可能也会很大,所以一遍一遍的模拟就会超时。我们发现,题中有一个要求:每次必须打离自己最近的活着的敌人。我们就只用枚举每个敌人即可,在枚举的过程中计算答案。细节处理这道题有点难度,因为$T$是$3$的倍数时能量会变成$3$。这是个周期问题,自然会想到除......
  • CF2050G Tree Destruction 题解
    【题意简述】你有一棵树,你可以从里面删除一条链上的节点,问剩下的点的联通块数量最大是多少。【思路】一眼树形dp,默认根为\(1\)。我们以这棵树的\(1\)节点作为示例。设\(dp[i][0]\)表示\(i\)节点的子树中选一条链,\(i\)​不在链上的最大联通块数。设\(dp[i][1]\)......
  • CF2050F Maximum modulo equality 题解
    【题意简述】你有一个长度为\(n\)的数组\(a\)。每一次询问给定\(l,r\),寻找最大的\(m\)使得\(a_l\)到\(a_r\)的所有数对\(m\)同余,【前置数学芝士】首先是一个非常Naive的结论,请自行感性证明:设\(a>b\),\(a\)和\(b\)对\(a-b\)同余。理性证明:设\(p=a-b\),\(......
  • P5007 DDOSvoid 的疑惑 题解
    题目传送门思路树形dp模版题。设\(dp_i\)为\(pos\)的最优解,\(dp2_i\)为只考虑\(pos\)子树时,毒瘤集的数量。可得:\(dp_i=dp_{i}\timesdp2_{son}+dp_{son}\timesdp2_{i}+dp_i+dp_{son}\)\(dp2_i=dp2_{i}\timesdp2_{son}+dp2_{i}+dp2_{son}\)用深搜来更新\(dp\)......
  • 题解:P1007 独木桥
    独木桥-洛谷https://www.luogu.com.cn/problem/P1007思路:输入部分:首先读取独木桥的长度 L 和初始留在桥上的士兵数目 N。然后通过循环读取每个士兵的初始坐标并存储在 soldiers 数组中。计算最小时间和最大时间:对于每个士兵,通过 min(soldiers[i],L+1-soldie......
  • 递归疑难问题解答
    1.计算一个数的每位之和(递归实现)写一个递归函数DigitSum(n),输入一个非负整数,返回组成它的数字之和例如,调用DigitSum(1729),则应该返回1+7+2+9,它的和是19输入:1729,输出:19#include<stdio.h>intfac(intn){ if(n<10) { returnn; } else { returnfac(n/10)+......