Codeforces Contest 1076 problem D Edge Deletion —— dijkstra的一些优化

2024-04-07 00:48

本文主要是介绍Codeforces Contest 1076 problem D Edge Deletion —— dijkstra的一些优化,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

You are given an undirected connected weighted graph consisting of n vertices and m edges. Let’s denote the length of the shortest path from vertex 1 to vertex i as di.

You have to erase some edges of the graph so that at most k edges remain. Let’s call a vertex i good if there still exists a path from 1 to i with length di after erasing the edges.

Your goal is to erase the edges in such a way that the number of good vertices is maximized.

Input
The first line contains three integers n, m and k (2≤n≤3⋅105, 1≤m≤3⋅105, n−1≤m, 0≤k≤m) — the number of vertices and edges in the graph, and the maximum number of edges that can be retained in the graph, respectively.

Then m lines follow, each containing three integers x, y, w (1≤x,y≤n, x≠y, 1≤w≤109), denoting an edge connecting vertices x and y and having weight w.

The given graph is connected (any vertex can be reached from any other vertex) and simple (there are no self-loops, and for each unordered pair of vertices there exists at most one edge connecting these vertices).

Output
In the first line print e — the number of edges that should remain in the graph (0≤e≤k).

In the second line print e distinct integers from 1 to m — the indices of edges that should remain in the graph. Edges are numbered in the same order they are given in the input. The number of good vertices should be as large as possible.

Examples
inputCopy
3 3 2
1 2 1
3 2 1
1 3 3
outputCopy
2
1 2
inputCopy
4 5 2
4 1 8
2 4 1
2 1 3
3 4 9
3 1 5
outputCopy
2
3 2

题意:

给你n个点,m条边,你需要删除一些边使得剩下的边的数量小于等于k,并且我们规定,一个点是“好的”是在删边以后1到这个点的最短距离等于删边之前的最短距离,让你求出需要剩下哪些边使得“好的”的路最多。

题解:

dijkstra求出1到所有点的距离,之后从第一个点出发,看看与它相邻的点有哪些是之前就是最短的,然后放到队列里面,同时ans数组push进去这条边的id,对于这题有一些优化,首先最大的优化是这个:
dijkstra的时候不要用vis数组纪录,而是在pop的时候看这一个状态的点过来的最短路是否等于当前的最短路,如果不是就说明它之后有状态比它更优,还有就是不要把自定义结构体放到队列里,用pa会快,不要用map,直接将这条边的id放到结构体里,还有就是查询和dijkstra可以一起做。
分开做的情况:
在这里插入图片描述

#include<bits/stdc++.h>
using namespace std;
#define ll long long
#define pa pair<ll,int>
const int N=3e5+5;
const ll inf=1e17;
struct node
{int to,next,id;ll val;
}e[N*2];
int head[N],cnt;
void add(int x,int y,ll w,int id)
{e[cnt].to=y;e[cnt].next=head[x];e[cnt].val=w;e[cnt].id=id;head[x]=cnt++;
}
ll dis[N];
int vis[N],last[N];
vector<int>ans;
void dijkstra()
{priority_queue<pa>Q;dis[1]=0;Q.push({0,1});while(!Q.empty()){pa u=Q.top();Q.pop();if(-u.first!=dis[u.second])continue;for(int i=head[u.second];~i;i=e[i].next){int v=e[i].to;ll w=e[i].val;if(dis[v]>dis[u.second]+w){dis[v]=dis[u.second]+w;Q.push({-dis[v],v});}}}
}
void check(int x)
{priority_queue<pa>Q;dis[1]=0;Q.push({0,1});vis[1]=1;while(!Q.empty()&&x){pa u=Q.top();Q.pop();for(int i=head[u.second];~i;i=e[i].next){int v=e[i].to;ll w=e[i].val;if(vis[v])continue;if(dis[v]==dis[u.second]+w){Q.push({-dis[v],v});vis[v]=1;ans.push_back(e[i].id);x--;}if(x<=0)break;}}
}
int main()
{//freopen("in.txt","r",stdin);memset(head,-1,sizeof(head));for(int i=1;i<N;i++)dis[i]=inf;int x,y;ll w;int n,m,k;scanf("%d%d%d",&n,&m,&k);for(int i=1;i<=m;i++){scanf("%d%d%lld",&x,&y,&w);add(x,y,w,i),add(y,x,w,i);}dijkstra();check(k);printf("%d\n",ans.size());for(int i=0;i<ans.size();i++)printf("%d%c",ans[i],i==ans.size()-1?'\n':' ');return 0;
}

合起来的情况:就是加一个数组在搜的时候就记录最优解的id
在这里插入图片描述

#include<bits/stdc++.h>
using namespace std;
#define ll long long
#define pa pair<ll,int>
const int N=3e5+5;
const ll inf=1e17;
struct node
{int to,next,id;ll val;
}e[N*2];
int head[N],cnt;
void add(int x,int y,ll w,int id)
{e[cnt].to=y;e[cnt].next=head[x];e[cnt].val=w;e[cnt].id=id;head[x]=cnt++;
}
ll dis[N];
int vis[N],last[N];
vector<int>ans;
void dijkstra(int x)
{priority_queue<pa>Q;dis[1]=0;Q.push({0,1});while(!Q.empty()&&x){pa u=Q.top();Q.pop();if(-u.first!=dis[u.second])continue;if(last[u.second])x--,ans.push_back(last[u.second]);for(int i=head[u.second];~i;i=e[i].next){int v=e[i].to;ll w=e[i].val;if(dis[v]>dis[u.second]+w){dis[v]=dis[u.second]+w;Q.push({-dis[v],v});last[v]=e[i].id;}}}
}
int main()
{//freopen("in.txt","r",stdin);memset(head,-1,sizeof(head));for(int i=1;i<N;i++)dis[i]=inf;int x,y;ll w;int n,m,k;scanf("%d%d%d",&n,&m,&k);for(int i=1;i<=m;i++){scanf("%d%d%lld",&x,&y,&w);add(x,y,w,i),add(y,x,w,i);}dijkstra(k);printf("%d\n",ans.size());for(int i=0;i<ans.size();i++)printf("%d%c",ans[i],i==ans.size()-1?'\n':' ');return 0;
}

这篇关于Codeforces Contest 1076 problem D Edge Deletion —— dijkstra的一些优化的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/881257

相关文章

Vue3 的 shallowRef 和 shallowReactive:优化性能

大家对 Vue3 的 ref 和 reactive 都很熟悉,那么对 shallowRef 和 shallowReactive 是否了解呢? 在编程和数据结构中,“shallow”(浅层)通常指对数据结构的最外层进行操作,而不递归地处理其内部或嵌套的数据。这种处理方式关注的是数据结构的第一层属性或元素,而忽略更深层次的嵌套内容。 1. 浅层与深层的对比 1.1 浅层(Shallow) 定义

HDFS—存储优化(纠删码)

纠删码原理 HDFS 默认情况下,一个文件有3个副本,这样提高了数据的可靠性,但也带来了2倍的冗余开销。 Hadoop3.x 引入了纠删码,采用计算的方式,可以节省约50%左右的存储空间。 此种方式节约了空间,但是会增加 cpu 的计算。 纠删码策略是给具体一个路径设置。所有往此路径下存储的文件,都会执行此策略。 默认只开启对 RS-6-3-1024k

使用opencv优化图片(画面变清晰)

文章目录 需求影响照片清晰度的因素 实现降噪测试代码 锐化空间锐化Unsharp Masking频率域锐化对比测试 对比度增强常用算法对比测试 需求 对图像进行优化,使其看起来更清晰,同时保持尺寸不变,通常涉及到图像处理技术如锐化、降噪、对比度增强等 影响照片清晰度的因素 影响照片清晰度的因素有很多,主要可以从以下几个方面来分析 1. 拍摄设备 相机传感器:相机传

csu 1446 Problem J Modified LCS (扩展欧几里得算法的简单应用)

这是一道扩展欧几里得算法的简单应用题,这题是在湖南多校训练赛中队友ac的一道题,在比赛之后请教了队友,然后自己把它a掉 这也是自己独自做扩展欧几里得算法的题目 题意:把题意转变下就变成了:求d1*x - d2*y = f2 - f1的解,很明显用exgcd来解 下面介绍一下exgcd的一些知识点:求ax + by = c的解 一、首先求ax + by = gcd(a,b)的解 这个

poj 1502 MPI Maelstrom(单源最短路dijkstra)

题目真是长得头疼,好多生词,给跪。 没啥好说的,英语大水逼。 借助字典尝试翻译了一下,水逼直译求不喷 Description: BIT他们的超级计算机最近交货了。(定语秀了一堆词汇那就省略吧再见) Valentine McKee的研究顾问Jack Swigert,要她来测试一下这个系统。 Valentine告诉Swigert:“因为阿波罗是一个分布式共享内存的机器,所以它的内存访问

MySQL高性能优化规范

前言:      笔者最近上班途中突然想丰富下自己的数据库优化技能。于是在查阅了多篇文章后,总结出了这篇! 数据库命令规范 所有数据库对象名称必须使用小写字母并用下划线分割 所有数据库对象名称禁止使用mysql保留关键字(如果表名中包含关键字查询时,需要将其用单引号括起来) 数据库对象的命名要能做到见名识意,并且最后不要超过32个字符 临时库表必须以tmp_为前缀并以日期为后缀,备份

uva 10025 The ? 1 ? 2 ? ... ? n = k problem(数学)

题意是    ?  1  ?  2  ?  ...  ?  n = k 式子中给k,? 处可以填 + 也可以填 - ,问最小满足条件的n。 e.g k = 12  - 1 + 2 + 3 + 4 + 5 + 6 - 7 = 12 with n = 7。 先给证明,令 S(n) = 1 + 2 + 3 + 4 + 5 + .... + n 暴搜n,搜出当 S(n) >=

uva 10801(乘电梯dijkstra)

题意: 给几个电梯,电梯0 ~ n-1分别可以到达很多层楼。 换乘电梯需要60s时间。 问从0层到target层最小的时间。 解析: 将进入第0层的电梯60s也算上,最后减。 坑点是如果target为0输出0。 代码: #include <iostream>#include <cstdio>#include <cstdlib>#include <algori

hdu 3790 (单源最短路dijkstra)

题意: 每条边都有长度d 和花费p,给你起点s 终点t,要求输出起点到终点的最短距离及其花费,如果最短距离有多条路线,则输出花费最少的。 解析: 考察对dijkstra的理解。 代码: #include <iostream>#include <cstdio>#include <cstdlib>#include <algorithm>#include <cstrin

poj 3255 次短路(第k短路) A* + spfa 或 dijkstra

题意: 给一张无向图,求从1到n的次短路。 解析: A* + spfa 或者 dijkstra。 详解见上一题:http://blog.csdn.net/u013508213/article/details/46400189 本题,spfa中,stack超时,queue的效率最高,priority_queue次之。 代码: #include <iostream>#i