bzoj2654
WOJ#3696 tree
给你一个无向带权连通图,每条边是黑色或白色。让你求一棵最小权的恰好有need条白色边的生成树。题目保证有解。
第一行V,E,need分别表示点数,边数和需要的白色边数。接下来E行每行s,t,c,col表示这边的端点(点从0开始标号),边权,颜色(0白色1黑色)。
一行表示所求生成树的边权和。
给定一个无向带权连通图,其中有白边和黑边。求在恰好取need条白边的条件下的最小生成树。
首先我们想到用邻接表存图,Kruskal求最小生成树。
但此题还有一个特殊条件:生成树上必须有且仅有 n e e d need need条白边。所以为了系统、简便地控制白边的数量,我们在求最小生成树之前先给所有白边加上一定的权值 x x x( x x x的求法之后会讲到)。考虑到Kruskal的思路,易证得 x x x越大,白边条数越少,反之越多。
再在算法进行过程中,记录树上(并查集中)加入白边的条数 c n t cnt cnt,得出结果。此时求出的 s u m sum sum,乃是 ( a n s + x ∗ c n t ) (ans+x*cnt) (ans+x∗cnt),故减去即可得出答案。
现在再来考虑求出正确的x的过程。题目约定边权在 [ 0 , 100 ] [0,100] [0,100]以内,所以我们的枚举范围在 [ − 100 , 100 ] [-100,100] [−100,100]以内,嘛,为了保险起见取到 [ − 105 , 105 ] [-105,105] [−105,105]吧。所以也就是说……我们要求200多次最小生成树?有点恐怖。考虑到这200多次中有很多重复的,我们采用二分答案。
具体二分过程:如果 c n t > = n e e d cnt>=need cnt>=need,则函数 k r u s k a l ( ) kruskal() kruskal()为真,否则为伪。二分过程中,如果 k r u s k a l ( ) kruskal() kruskal()为真,则用 ( s u m − m i d ∗ c n t ) (sum-mid*cnt) (sum−mid∗cnt)去更新 a n s ans ans( m i d mid mid就是 x x x),否则不更新。这样,又一道难题也就被解决了。
作几点说明:
#include
#include
#include
using namespace std;
const int maxm=1e5+10;
int n,m,need,l,r,cnt,tot,ans;
int u[maxm],v[maxm],w[maxm],c[maxm];
int fa[maxm];
struct edge {int u,v,w,c;};
edge e[maxm];
int read()
{
int x=0,f=1;char ch=getchar();
while(ch<'0'||ch>'9'){if(ch=='-')f=-1;ch=getchar();}
while(ch>='0'&&ch<='9'){x=x*10+ch-'0';ch=getchar();}
return x*f;
}
inline bool cmp(const edge &a,const edge &b)
{
return a.w==b.w?a.c<b.c:a.w<b.w;
}
inline int get(int x)
{
return x==fa[x]?x:fa[x]=get(fa[x]);
}
inline bool kruskal(int x)
{
tot=0,cnt=0;
int f1,f2,sum=0;
for(int i=1;i<=n;i++) fa[i]=i;
for(int i=1;i<=m;i++)
{
e[i].u=u[i],e[i].v=v[i],e[i].w=w[i],e[i].c=c[i];
if(!c[i]) e[i].w+=x;//给所有白边加上一定的权值x
}
sort(e+1,e+m+1,cmp);
for(int i=1;i<=m;i++)
{
f1=get(e[i].u),f2=get(e[i].v);
if(f1!=f2)
{
fa[f1]=f2;
sum++;
tot+=e[i].w;
if(!e[i].c) cnt++;//记录白边数量
}
if(sum==n-1) return cnt>=need;
}
}
int main()
{
n=read(),m=read(),need=read();
for(int i=1;i<=m;i++)
u[i]=read()+1,v[i]=read()+1,w[i]=read(),c[i]=read();//离线操作
l=-105,r=105;
while(l<r)
{
int mid=(l+r)>>1;
if(kruskal(mid)) l=mid+1,ans=tot-need*mid;
else r=mid;
}//二分答案
cout<<ans;
return 0;
}