• [BZOJ2654]tree(二分+Kruskal)


    2654: tree

    Time Limit: 30 Sec  Memory Limit: 512 MB
    Submit: 2733  Solved: 1124
    [Submit][Status][Discuss]

    Description

    给你一个无向带权连通图,每条边是黑色或白色。让你求一棵最小权的恰好有need条白色边的生成树。
    题目保证有解。

    Input

    第一行V,E,need分别表示点数,边数和需要的白色边数。
    接下来E行,每行s,t,c,col表示这边的端点(点从0开始标号),边权,颜色(0白色1黑色)。

    Output

    一行表示所求生成树的边权和。
    V<=50000,E<=100000,所有数据边权为[1,100]中的正整数。

    Sample Input

    2 2 1
    0 1 1 1
    0 1 2 0

    Sample Output

    2

    HINT

    原数据出错,现已更新 by liutian,但未重测---2016.6.24

    Source

     
    [Submit][Status][Discuss]

    一种叫WQS二分的思想,据说[九省联考2018]林克卡特树用到了这个东西。

    tsinsen.com/resources/Train2012-sol-wqs.pdf

    但是这道题不看论文也可以直接做,将每条白边加上x后求MST,设树上的白边的个数为f(x),可以确定f(x)是单调不增的,二分即可。

    但可能f(mid)>k,f(mid+1)<k,我们把相同长度的白边放在黑边的前面即可。

    https://www.cnblogs.com/NaVi-Awson/p/7252243.html

     1 #include<cstdio>
     2 #include<algorithm>
     3 #define rep(i,l,r) for (int i=l; i<=r; i++)
     4 typedef long long ll;
     5 using namespace std;
     6 
     7 const int N=100100;
     8 int n,m,cnt,tot,k,ans,u[N],v[N],w[N],c[N],fa[N];
     9 struct E{ int u,v,w,c; }e[N];
    10 
    11 bool operator<(E a,E b){ return a.w==b.w ? a.c<b.c : a.w<b.w; }
    12 int find(int x){ return x==fa[x] ? x : fa[x]=find(fa[x]); }
    13 
    14 bool check(int x){
    15     tot=cnt=0;
    16     rep(i,1,n) fa[i]=i;
    17     rep(i,1,m){
    18         e[i].u=u[i]; e[i].v=v[i]; e[i].w=w[i]; e[i].c=c[i];
    19         if(!c[i])e[i].w+=x;
    20     }
    21     sort(e+1,e+m+1);
    22     rep(i,1,m){
    23         int p=find(e[i].u),q=find(e[i].v);
    24         if(p!=q){
    25             fa[p]=q; tot+=e[i].w;
    26             if (!e[i].c) cnt++;
    27         }
    28     }
    29     return cnt>=k;
    30 }
    31 
    32 int main(){
    33     scanf("%d%d%d",&n,&m,&k);
    34     rep(i,1,m) scanf("%d%d%d%d",&u[i],&v[i],&w[i],&c[i]),u[i]++,v[i]++;
    35     int L=-105,R=105;
    36     while(L<=R){
    37         int mid=(L+R)>>1;
    38         if(check(mid)) L=mid+1,ans=tot-k*mid; else R=mid-1;
    39     }
    40     printf("%d
    ",ans);
    41     return 0;
    42 }
  • 相关阅读:
    杭电1827
    hdu 3118
    poj 2060
    hdu 2236
    poj 2226
    poj 1719
    poj 1466
    poj 3160
    骑士飞行棋笔记
    基础测试学习笔记
  • 原文地址:https://www.cnblogs.com/HocRiser/p/8809347.html
Copyright © 2020-2023  润新知