zoukankan      html  css  js  c++  java
  • bzoj4519: [Cqoi2016]不同的最小割

    http://www.lydsy.com/JudgeOnline/problem.php?id=4519

    最小割树

    #include<queue>
    #include<cstdio>
    #include<cstring>
    #include<iostream>
    #include<algorithm>
     
    using namespace std;
     
    #define N 900
    #define M 9000
    const int inf=2e9;
     
    int n;
     
    int tot=1;
    int front[N],nxt[M<<1],to[M<<1],val[M<<1],from[M<<1];
    int lev[N],num[N];
    int path[N];
    int cur[N];
     
    int src,decc;
     
    int a[N],tmp[N];
    bool use[N];
     
    int dis[N][N];
    
    int ans[361350];
     
    void read(int &x)
    {
        x=0; char c=getchar();
        while(!isdigit(c))  c=getchar(); 
        while(isdigit(c)) { x=x*10+c-'0'; c=getchar();  }
    }
     
    void add(int u,int v,int w)
    {
        to[++tot]=v; nxt[tot]=front[u]; front[u]=tot; from[tot]=u; val[tot]=w;
    }
     
    bool bfs()
    {
        queue<int>q;
        for(int i=1;i<=n;++i) lev[i]=n;
        q.push(decc);
        lev[decc]=0;
        int now,t;
        while(!q.empty())
        {
            now=q.front();
            q.pop();
            for(int i=front[now];i;i=nxt[i])
            {
                t=to[i];
                if(lev[t]==n && val[i^1]) 
                {
                    lev[t]=lev[now]+1;
                    q.push(t);
                }
            }
        }
        return lev[src]!=n;
    }
     
    int augment()
    {
        int now=decc,flow=inf;
        int i;
        while(now!=src)
        {
            i=path[now];
            flow=min(flow,val[i]);
            now=from[i];
        }
        now=decc;
        while(now!=src)
        {
            i=path[now];
            val[i]-=flow;
            val[i^1]+=flow;
            now=from[i];
        }
        return flow;
    }
     
    int isap()
    {
        int flow=0;
        if(!bfs()) return 0;
        memset(num,0,sizeof(num));
        for(int i=1;i<=n;++i) num[lev[i]]++,cur[i]=front[i];
        int now=src,t;
        while(lev[src]<n)
        {
            if(now==decc)
            {
                flow+=augment();
                now=src;
            }
            bool advanced=false;
            for(int i=cur[now];i;i=nxt[i])
            {
                t=to[i];
                if(lev[t]==lev[now]-1 && val[i])
                {
                    advanced=true;
                    path[t]=i;
                    cur[now]=i;
                    now=t;
                    break;
                }
            }
            if(!advanced)
            {
                int mi=n;
                for(int i=front[now];i;i=nxt[i])
                    if(val[i]) mi=min(mi,lev[to[i]]);
                if(!--num[lev[now]]) break;
                num[lev[now]=mi+1]++;
                cur[now]=front[now];
                if(now!=src) now=from[path[now]];
            }
        }
        return flow;
    }
     
    void dfs(int x)
    {
        use[x]=true;
        for(int i=front[x];i;i=nxt[i])
            if(!use[to[i]] && val[i]) dfs(to[i]);
    }
     
    void solve(int l,int r)
    {
        if(l==r) return;
        for(int i=2;i<=tot;i+=2) val[i]=val[i+1]=val[i]+val[i+1]>>1;
        src=a[l]; decc=a[r];
        int flow=isap();
        memset(use,false,sizeof(use));
        dfs(src);
        for(int i=1;i<=n;++i)
            if(use[i]) 
                for(int j=1;j<=n;++j)
                    if(!use[j])
                        dis[i][j]=dis[j][i]=min(dis[i][j],flow);
        int i=l,j=r;
        for(int k=l;k<=r;++k) 
            if(use[a[k]]) tmp[i++]=a[k];
            else tmp[j--]=a[k];
        for(int k=l;k<=r;++k) a[k]=tmp[k];
        solve(l,i-1);
        solve(j+1,r);
    }
     
    int main()
    {
        int m;
        int u,v,w;
        memset(dis,127,sizeof(dis));
        read(n); read(m);
        for(int i=1;i<=n;++i) a[i]=i;
        while(m--)
        {
            read(u); read(v); read(w);
            add(u,v,w);
            add(v,u,w);
        }
        solve(1,n);
        int sum=0;
        for(int i=1;i<=n;++i)
            for(int j=i+1;j<=n;++j)
                ans[++sum]=dis[i][j];
        sort(ans+1,ans+sum+1);
        sum=unique(ans+1,ans+sum+1)-ans-1;
        cout<<sum;
        return 0;
    }

    4519: [Cqoi2016]不同的最小割

    Time Limit: 20 Sec  Memory Limit: 512 MB
    Submit: 866  Solved: 499
    [Submit][Status][Discuss]

    Description

    学过图论的同学都知道最小割的概念:对于一个图,某个对图中结点的划分将图中所有结点分成
    两个部分,如果结点s,t不在同一个部分中,则称这个划分是关于s,t的割。对于带权图来说,将
    所有顶点处在不同部分的边的权值相加所得到的值定义为这个割的容量,而s,t的最小割指的是在
    关于s,t的割中容量最小的割。
    而对冲刺NOI竞赛的选手而言,求带权图中两点的最小割已经不是什么难事了。我们可以把
    视野放宽,考虑有N个点的无向连通图中所有点对的最小割的容量,共能得到N(N−1)
    2个数值。
    这些数值中互不相同的有多少个呢?这似乎是个有趣的问题。
     

    Input

    输入文件第一行包含两个数N,M,表示点数和边数。接下来M行,每行三个数u,v,w,
    表示点u和点v(从1开始标号)之间有条边权值是w。
    1<=N<=850 1<=M<=8500 1<=W<=100000
     

    Output

     输出文件第一行为一个整数,表示个数。

     

    Sample Input

    4 4
    1 2 3
    1 3 6
    2 4 5
    3 4 4

    Sample Output

    3
  • 相关阅读:
    uva 10129 play on words——yhx
    uva 10305 ordering tasks(超级烂题)——yhx
    uva 816 abbott's revenge ——yhx
    php 格式化数字 位数不足前面加0补足
    NetBeans-xdebug的安装
    php 文件file常用的操作
    pg 匹配中文字符
    隐藏 php apache 的版本号
    php redis扩展
    php 二维数组的排序
  • 原文地址:https://www.cnblogs.com/TheRoadToTheGold/p/8042852.html
Copyright © 2011-2022 走看看