Codeforces GYM 100876 J - Buying roads 题解

才不是因为有了图床来测试一下呢,哼(

题意

给你\(N\)个点,\(M\)条带权边的无向图,选出\(K\)条边,使得得到的子图联通并且总代价最小,输出最小总代价和一种方案。

虽然题目里描述的很冗长,但其实这个图有一些性质:它最初是一条链/一个环,然后再有一些结点直接连到这些在链上/环上的结点。、

下图就是一个(就是样例):

做法

首先我们可以简单的查看点的度数来找到链/环上的点,和连接它们的边。

然后我们可以通过对链的头和尾添加一条权值大到足够保证不会选到它的边,把链变成环(样例被更改的样子):

我们可以枚举选择的是环上的相连的一条链,选择这些环上的边的同时,也贪心地从小到大选择挂在环上(类似于连接\(8\)和\(9\))的边。

这里可以枚举链的起点,在不断扩展这条链的右端点(并强制选取达到的它的边)的同时,维护一个堆,堆里存放的是选择的链上挂的边的权值最大值,每次添加时去查看添加的边的权值是否小于这个最大值,然后更新这个堆即可。

需要注意有选择整个环而非一部分的链的方案。

程序

这里把计算答案和构造方案分开了,实现起来可能比较方便。

#include<bits/stdc++.h>
using namespace std; typedef long long ll; int n,m,K;
vector<pair<int,int>> g[2005];
vector<int> hang[2005];
vector<int> node,edge;
ll w[2005],ans=1e18; void get_imp(const int &x,const int &p,const int &head){
node.emplace_back(x);
for(pair<int,int> &e:g[x]){
if(e.first!=p&&g[e.first].size()>1){
edge.emplace_back(e.second);
if(e.first!=head)get_imp(e.first,x,head);
return;
}
}
} void get_ans_val(){
priority_queue<int> heap;
ll cur;
for(int i=0;i<node.size();i++){
while(!heap.empty())heap.pop();
cur=0;
for(int jj=i;jj<i+K&&jj<i+node.size();jj++){
int j=jj%node.size();
for(int k=0;k<hang[node[j]].size();k++){
if(heap.size()+jj-i<K){
heap.emplace(w[hang[node[j]][k]]);
cur+=w[hang[node[j]][k]];
}else if(heap.top()>w[hang[node[j]][k]]){
cur-=heap.top();
heap.pop();
cur+=w[hang[node[j]][k]];
heap.emplace(w[hang[node[j]][k]]);
}else break;
}
if(heap.size()+jj-i==K){
ans=min(ans,cur);
cur-=heap.top();
heap.pop();
}
cur+=w[edge[j]];
}
ans=min(ans,cur);
}
} void construct(set<int> &use){
priority_queue<pair<int,int>> heap;
ll cur;
for(int i=0;i<node.size();i++){
while(!heap.empty())heap.pop();
cur=0;
use.clear();
for(int jj=i;jj<i+K&&jj<i+node.size();jj++){
int j=jj%node.size();
for(int k=0;k<hang[node[j]].size();k++){
if(heap.size()+jj-i<K){
heap.emplace(w[hang[node[j]][k]],hang[node[j]][k]);
use.insert(hang[node[j]][k]);
cur+=w[hang[node[j]][k]];
}else if(heap.top().first>w[hang[node[j]][k]]){
cur-=heap.top().first;
use.erase(heap.top().second);
heap.pop();
cur+=w[hang[node[j]][k]];
heap.emplace(w[hang[node[j]][k]],hang[node[j]][k]);
use.insert(hang[node[j]][k]);
}else break;
}
if(heap.size()+jj-i==K){
if(cur==ans)return;
cur-=heap.top().first;
use.erase(heap.top().second);
heap.pop();
}
cur+=w[edge[j]];
use.insert(edge[j]);
}
if(cur==ans)return;
}
} int main(){ ios::sync_with_stdio(false);
cin.tie(0);
cout.tie(0); if(fopen("roads.in","r")){
freopen("roads.in","r",stdin);
freopen("roads.out","w",stdout);
} cin>>n>>m>>K;
for(int i=1;i<=m;i++){
static int u,v,W;
cin>>u>>v>>W;
g[u].emplace_back(v,i);
g[v].emplace_back(u,i);
w[i]=W;
}
for(int i=1;i<=n;i++){
if(g[i].size()==1)hang[g[i].front().first].emplace_back(g[i].front().second);
else if(node.empty()){
int cnt=0;
for(pair<int,int> &e:g[i])cnt+=g[e.first].size()>1;
if(cnt==1)get_imp(i,-1,i);
}
}
if(node.empty())for(int i=1;i<=n;i++){
if(g[i].size()>1){
get_imp(i,-1,i);
break;
}
}
if(edge.size()+1==node.size()){
edge.emplace_back(++m);
w[m]=1e15;
}
for(int i=1;i<=n;i++)sort(hang[i].begin(),hang[i].end(),[](const int &a,const int &b){
return w[a]<w[b];
});
get_ans_val();
cout<<ans<<'\n';
set<int> s;
construct(s);
for(const int &x:s)cout<<x<<'\n'; return 0;
}

Codeforces GYM 100876 J - Buying roads 题解的更多相关文章

  1. Codeforces gym 101343 J.Husam and the Broken Present 2【状压dp】

     2017 JUST Programming Contest 2.0 题目链接:Codeforces gym 101343 J.Husam and the Broken Present 2 J. Hu ...

  2. codeforces Gym 100187J J. Deck Shuffling dfs

    J. Deck Shuffling Time Limit: 2   Sec Memory Limit: 256 MB 题目连接 http://codeforces.com/gym/100187/pro ...

  3. codeforces GYM 100114 J. Computer Network 无相图缩点+树的直径

    题目链接: http://codeforces.com/gym/100114 Description The computer network of “Plunder & Flee Inc.” ...

  4. codeforces Gym 100500 J. Bye Bye Russia

    Problem J. Bye Bye RussiaTime Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://codeforces.com/gym/1005 ...

  5. codeforces GYM 100114 J. Computer Network tarjan 树的直径 缩点

    J. Computer Network Time Limit: 1 Sec Memory Limit: 256 MB 题目连接 http://codeforces.com/gym/100114 Des ...

  6. Codeforces Gym 100338C C - Important Roads tarjan

    C - Important RoadsTime Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://acm.hust.edu.cn/vjudge/contes ...

  7. codeforces gym 100947 J. Killing everything dp+二分

    J. Killing everything time limit per test 4 seconds memory limit per test 64 megabytes input standar ...

  8. codeforces gym 100357 J (网络流)

    题目大意 有n种物品,m种建筑,p个人. n,m,p∈[1,20] 每种建筑需要若干个若干种物品来建造.每个人打算建造一种建筑,拥有一些物品. 主角需要通过交易来建造自己的建筑,交易的前提是对方用多余 ...

  9. Codeforces Gym 100114 J. Computer Network

    Description 给出一个图,求添加一条边使得添加后的图的桥(割边)最少. Sol Tarjan. 一遍Tarjan求割边. 我们发现连接的两个点一定是这两个点之间的路径上的桥最多,然后就可以贪 ...

随机推荐

  1. [atARC098F]Donation

    贪心,一定在最后一次经过某节点时付出$b_{u}$,条件是付出后$W\ge \max(a_{i}-b_{i},0)$(同时也可以仅考虑这个限制,因为$W$在过程中不会增大) 假设"最后一次经 ...

  2. [bzoj3170]松鼠聚会

    这个距离就是切比雪夫距离,有一个神奇的东西是说将(x,y)变成(x+y,x-y),然后就是曼哈顿距离,因此转化后对x坐标和y坐标分别统计排序和求和(求前缀和预处理+二分) 1 #include< ...

  3. Java 中的监控与管理原理概述

    点赞再看,动力无限.Hello world : ) 微信搜「程序猿阿朗 」. 本文 Github.com/niumoo/JavaNotes 和 程序猿阿朗博客 已经收录,有很多知识点和系列文章. 当前 ...

  4. CF1610F F. Mashtali: a Space Oddysey

    我们首先发现有如下性质: 我们不妨先随机定向边,那么我们发现无论我们如何翻转边. 都会对其两端的点,造成 \(2 / 4\) 的影响,所以我们发现如果一个点其和他相连的所有边权和为偶数,则我们不能调整 ...

  5. Codeforces 497E - Subsequences Return(矩阵乘法)

    Codeforces 题目传送门 & 洛谷题目传送门 一道还算不错的矩乘 tea 罢,不过做过类似的题应该就比较套路了-- 首先考虑对于一个固定的序列 \(\{a\}\) 怎样求其本质不同的序 ...

  6. pyquery解析库的介绍和使用

    ### pyquery的介绍和使用 ## 测试文本 text = ''' <html><head><title>there is money</title&g ...

  7. Python函数之传参

    目录 1. 函数传参 1.1 参数的作用 1.2 形参和实参 1.3 位置参数 1.4 关键字参数 1.5 默认实参 1.6 参数总结 2. 可变参数 1. 函数传参 1.1 参数的作用 1.2 形参 ...

  8. 48-Merge Sorted Array

    $88. Merge Sorted Array My Submissions QuestionEditorial Solution Total Accepted: 98885 Total Submis ...

  9. EXCEL excel中运用ctrl+D、ctrl+enter、ctrl+E批量填充数据

    在excel中,利用鼠标拖动可以快速向下或者向右填充序列或者复制单元格.但是利用快捷键也可以实现多种填充方式,本文就为大家介绍一些ctrl系列快捷键的填充,一起来看看吧. 封面tu 一,ctrl+D/ ...

  10. 用C语言的LED实验,有汇编哦!

    C语言LED实验 1.汇编激活CPU 首先要明白对于没有系统开发板(也就是裸机)来说,是没办法直接对C进行识别.所以需要一段汇编语言,来配置CPU的资源,选择CPU运行模式,初始化指针位置. 代码如下 ...