首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >专栏 >Educational Codeforces Round 54 (Rated for Div. 2) D. Edge Deletion(dijkstra+bfs)

Educational Codeforces Round 54 (Rated for Div. 2) D. Edge Deletion(dijkstra+bfs)

作者头像
Ch_Zaqdt
发布2019-01-10 16:10:49
发布2019-01-10 16:10:49
39200
代码可运行
举报
文章被收录于专栏:Zaqdt_ACMZaqdt_ACM
运行总次数:0
代码可运行

题目链接:http://codeforces.com/contest/1076/problem/D

       题意是输入n,m,k,表示有n个点,m条边,需要保留k条边,现在要保证1-n的最短距离不变,然后进行删边,要至少保留k条边且使包含的节点数最多。

       思路就是先用dij跑一遍1-n的最短路(spfa会被卡),然后再用dij的思想去跑一遍bfs,把最短路的边都存下来,其实就是一个记录路径的过程,然后输出前k个。这道题的坑点很多,因为权值是1e9的所以要开ll,然后对于inf也要初始化为1e18,还有我用的链式前向星,存边的话数组要开二倍,不然会RE on 7(别问我怎么知道的),其他的应该就没什么了,主要就是细节需要细心点...


AC代码:

代码语言:javascript
代码运行次数:0
运行
复制
#include <bits/stdc++.h>
#define maxn 300005
#define inf 1e18
#define ll long long
using namespace std;
struct Node{
	int to,next,id;
	ll w;
	bool operator < (const Node &a) const {
		return a.w < w;
	}
}Edge[maxn << 1],Now,Next;
int head[maxn],num;
ll dist[maxn];
bool vis[maxn];
vector<int> v;
int n,m,k;

void init(){
	memset(head,-1,sizeof(head));
	num = 0;
}

void add(int u, int v,ll w,int id){
	Edge[num].to = v;
	Edge[num].w = w;
	Edge[num].id = id;
	Edge[num].next = head[u];
	head[u] = num ++;
}

void dijkstra(){
	memset(vis,false,sizeof(vis));
	for(int i=1;i<=n;i++) dist[i] = inf;
	priority_queue<Node> q;
	Now.to = 1;
	dist[1] = 0;
	q.push(Now);
	while(!q.empty()){
		Now = q.top();
		q.pop();
		int ans = Now.to;
		if(vis[ans]) continue;
		vis[ans] = true;
		for(int i=head[ans];i!=-1;i=Edge[i].next){
			int to = Edge[i].to;
			if(!vis[to] && dist[to] > dist[ans] + Edge[i].w){
				dist[to] = dist[ans] + Edge[i].w;
				Next.to = to;
				Next.w = dist[to];
				q.push(Next);
			}
		}
	}
}

void solve(){
	memset(vis,false,sizeof(vis));
	vis[1] = true;
	queue<int> q;
	q.push(1);
	while(!q.empty()){
		int ans = q.front();
		q.pop();
		for(int i=head[ans];i!=-1;i=Edge[i].next){
			int to = Edge[i].to;
			if(!vis[to] && dist[to] == dist[ans] + Edge[i].w){
				vis[to] = true;
				dist[to] = dist[ans] + Edge[i].w;
				v.push_back(Edge[i].id);
				q.push(to);
			}
		}
	}
}

int main()
{
	init();
	scanf("%d%d%d",&n,&m,&k);
	for(int i=1;i<=m;i++){
		int u,v;
		ll w;
		scanf("%d%d%lld",&u,&v,&w);
		add(u, v, w, i);
		add(v, u, w, i);
	}
	dijkstra();
	solve();
	int cnt = v.size();
	if(cnt > k) cnt = k;
	printf("%d\n",cnt);
	for(int i=0;i<cnt;i++){
		printf("%d%c",v[i],i==cnt-1?'\n':' ');
	}
	return 0;
}
本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2018年11月14日,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档