给你一个无向带权连通图,每条边是黑色或白色。让你求一棵最小权的恰好有need条白色边的生成树。
题目保证有解。
题解: orz这个二分 首先我们来分析一下 边数与权值的关系 显然的随着白色边的权值增加会导致白色边的数量减少那么 我们可以通过二分找到恰好为need的白边的位置 同同构造MST保证此时的边权和最小 出现的问题是你二分加的权值会出现白边和黑边相等情况 此时优先白边加入 排序的时候加一下限制即可
#include <bits/stdc++.h> const int MAXN=5e4+10; using namespace std; int n,m,ned; typedef struct node{ int id,u,v,vul,key; }node; node d[MAXN<<1]; bool cmp(node aa,node bb){ if(aa.key==bb.key)return aa.id<bb.id; return aa.key<bb.key; } int f[MAXN],key; int find1(int x){ if(f[x]==x)return x; else return f[x]=find1(f[x]); } bool check(int t){ for(int i=1;i<=m;i++){ if(d[i].id==0)d[i].key=d[i].vul+t; else d[i].key=d[i].vul; } sort(d+1,d+m+1,cmp);key=0; int cnt=0; for(int i=1;i<=n;i++)f[i]=i; for(int i=1;i<=m;i++){ int t1=find1(d[i].u);int t2=find1(d[i].v); if(t1==t2)continue; f[t1]=t2;key+=d[i].key; if(!d[i].id)cnt++; } if(cnt>=ned)return 1; return 0; } int main(){ scanf("%d%d%d",&n,&m,&ned); for(int i=1;i<=n;i++)f[i]=i; for(int i=1;i<=m;i++)scanf("%d%d%d%d",&d[i].u,&d[i].v,&d[i].vul,&d[i].id),d[i].u++,d[i].v++; int l=-110,r=110,ans; while(l<=r){ int mid=(l+r)>>1; if(check(mid))l=mid+1,ans=key-ned*mid; else r=mid-1; } printf("%d\n",ans); }
原数据出错,现已更新 by liutian,但未重测---2016.6.24
原文:https://www.cnblogs.com/wang9897/p/9461808.html