最短路径树,CF 1076D - Edge Deletion

目录

一、题目

1、题目描述

2、输入输出

2.1输入

2.2输出

3、原题链接

二、解题报告

1、思路分析

2、复杂度

3、代码详解


一、题目

1、题目描述

2、输入输出

2.1输入

2.2输出

3、原题链接

1076D - Edge Deletion


二、解题报告

1、思路分析

考虑dijkstra实际上是利用贪心算法求了一棵最短路径树,详见:最短路合集,Dijkstra,堆优化Dijkstra,BellmanFord,SPFA,Floyd,附完整代码及OJ链接-CSDN博客

最短路径树中有n - 1条边,如果k > n - 1,那么我们保留n - 1条树边即可

否则,保留离源点最近的k个结点的前驱边

2、复杂度

时间复杂度: O(MlogM + nlogn)空间复杂度:O(N + M)

3、代码详解

 ​
#include <bits/stdc++.h>
#include <ranges>

using i64 = long long;
using i32 = unsigned int;
using u64 = unsigned long long;
using i128 = __int128;

constexpr int inf32 = 1E9 + 7;
constexpr i64 inf64 = 1E18 + 7;
constexpr int P = 1'000'000'007;

void solve() {
	int n, m, k;
	std::cin >> n >> m >> k;

	std::vector<std::vector<int>> g(n);
	std::vector<std::array<int, 2>> edges;
	
	for (int i = 0, u, v, w; i < m; ++ i) {
		std::cin >> u >> v >> w;
		-- u, -- v;
		g[u].push_back(edges.size());
		edges.push_back({v, w});
		g[v].push_back(edges.size());
		edges.push_back({u, w});	
	}

	std::vector<i64> dst(n, inf64);
	std::vector<int> pre(n, -1);

	std::priority_queue<std::pair<i64, int>, std::vector<std::pair<i64, int>>, std::greater<std::pair<i64, int>>> pq;

	pq.emplace(dst[0] = 0, 0);

	while (pq.size()) {
		auto [d, u] = pq.top();
		pq.pop();

		if (d > dst[u]) continue;

		for (int i : g[u]) {
			auto [v, w] = edges[i];
			if (d + w < dst[v]) {
				pre[v] = i;
				pq.emplace(dst[v] = d + w, v);
			}
		}
	}

	int ans = std::min(k, n - 1);

	std::cout << ans << '\n';

	std::vector<int> st(n - 1);
	std::iota(st.begin(), st.end(), 1);

	std::ranges::sort(st, [&dst](int u, int v) -> bool{
		return dst[u] < dst[v];
	});

	for (int i = 0; i < ans; ++ i) {
		std::cout << pre[st[i]] / 2 + 1 << " \n"[i + 1 == ans];
	}
}

auto FIO = []{
	std::ios::sync_with_stdio(false);
	std::cin.tie(nullptr);
	std::cout.tie(nullptr);
	return 0;
}();

int main () {
	#ifdef DEBUG
		freopen("in.txt", "r", stdin);
		freopen("out.txt", "w", stdout);
	#endif
	
	int T = 1;
	// std::cin >> T;
	while (T --)
		solve();

	return 0;
}

  • 4
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

EQUINOX1

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值