首页 > 其他 > 详细

[CF1076D] Edge Deletion - Dijkstra,最短路径树

时间:2020-03-27 21:31:13      阅读:53      评论:0      收藏:0      [点我收藏+]

给定无向带权连通图,保留至多 \(k\) 条边,最大化到 \(1\) 号节点最短路长度不变的点的数量。

Solution

一个显然的做法是,构建原图的一棵最短路径树,任意选择一个大小为 \(k\) 的包含根的连通块就是答案

另一方面,我们回归到 Dijkstra 算法的原理,不难发现,我们只需要在算法加了 \(k\) 条边以后停止,当前选择的边集就是答案

#include <bits/stdc++.h>
using namespace std;
#define reset3f(x) memset(x,0x3f,sizeof x)
#define int long long
const int N = 1000005;

int n,m,k,t1,t2,t3;
struct pii {
    int first,second,third;
};
map<int,int> mp;
namespace sp {
const int N=1e+6+5;
vector<pii> g[N];
int n,v0=1,d[N],pre[N];
void make(int t1,int t2,int t3,int t4) {
    g[t1].push_back({t2,t3,t4});
    g[t2].push_back({t1,t3,t4});
}
void reset_graph() {
    for(int i=0;i<=n;i++) g[i].clear();
}
void solve() {
    priority_queue<pair<int,int> > qu;
    reset3f(d);
    d[v0]=0;
    qu.push(make_pair(0,v0));
    while(qu.size()) {
        int p=qu.top().second,r=qu.top().first;
        qu.pop();
        if(r+d[p]) continue;
        if(pre[p]) mp[pre[p]]++;
        if(mp.size()>=k) {
            for(auto i=mp.begin();i!=mp.end();i++)
                cout<<i->first<<" ";
            exit(0);
        }
        for(int i=0;i<g[p].size();i++) {
            int q=g[p][i].first,w=g[p][i].second;
            if(d[q]>d[p]+w) {
                d[q]=d[p]+w;
                qu.push(make_pair(-d[q],q));
                pre[q]=g[p][i].third;
            }
        }
    }
}
}

signed main() {
    ios::sync_with_stdio(false);
    cin>>n>>m>>k;
    k=min(k,n-1);
    cout<<k<<endl;
    for(int i=1;i<=m;i++) {
        cin>>t1>>t2>>t3;
        sp::make(t1,t2,t3,i);
    }
    sp::solve();
}

[CF1076D] Edge Deletion - Dijkstra,最短路径树

原文:https://www.cnblogs.com/mollnn/p/12584191.html

(0)
(0)
   
举报
评论 一句话评论(0
关于我们 - 联系我们 - 留言反馈 - 联系我们:wmxa8@hotmail.com
© 2014 bubuko.com 版权所有
打开技术之扣,分享程序人生!