前言

摆王兴致冲冲地跑到我们机房来对我说跟你讲一个黑科技。。。

Dinic的神奇优化

Dinic优化

我们发现如果Dinic不建反向边会跑的飞起(当然Wa是必然的)
所以考虑在加反向边的基础上优化.

首先我们记录网络中最大的一个流量,设它为Min,然后:

  1. 把所有小于Min的边都加入网络中
  2. 最大流+=Dinic()
  3. Min /= 2
  4. 到1
    然后在Dinic时不走反向边(但是值要改变),最后在可以走反向边的情况下再来一次

注意bfs的一些操作,如果不当会溢出...

代码实现

#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<math.h>
#include<algorithm>
#include<queue>
#include<set>
#include<map>
#include<iostream>
using namespace std;
#define ll long long
#define re register
#define file(a) freopen(a".in","r",stdin);freopen(a".out","w",stdout)
#define int ll
inline int gi()
{
    int f=1,sum=0;char ch=getchar();
    while(ch>'9' || ch<'0'){if(ch=='-')f=-1;ch=getchar();}
    while(ch>='0' && ch<='9'){sum=(sum<<3)+(sum<<1)+ch-'0';ch=getchar();}
    return f*sum;
}
const int N=500010,M=2000010,Inf=1e10+10;
int n,m,s,t,ans,dep[N],cur[N];
struct node
{
    int u,v,c;
}E[M];
vector<int>front[N];
vector<node>e;
void Add(int u,int v,int c)
{
    e.push_back((node){u,v,c});
    e.push_back((node){v,u,0});
    front[u].push_back(e.size()-2);
}
bool cmp(node a,node b)
{
    return a.c>b.c;
}
queue<int>Q;
bool bfs()
{
    memset(dep,127,sizeof(dep));
    Q.push(s);dep[s]=0;
    while(!Q.empty())
    {
        int u=Q.front();Q.pop();
        for(int i=0;i<front[u].size();i++)
        {
            int id=front[u][i];
            int v=e[id].v;
            if(dep[v]>1e9 && e[id].c)
            {
                dep[v]=dep[u]+1;
                Q.push(v);
            }
        }
    }
    return dep[t]<1e9;
}
int dfs(int u,int flow)
{
//  printf("%lld %lld\n",u,flow);
    if(!flow || u==t)return flow;
    int F=0;
    for(int &i=cur[u];i<front[u].size();i++)
    {
        int id=front[u][i];
        if(dep[e[id].v]==dep[u]+1 && e[id].c)
        {
            int di=dfs(e[id].v,min(flow,e[id].c));
            if(di)
            {
                e[id].c-=di;e[id^1].c+=di;
                flow-=di;F+=di;
                if(!flow)break;
            }
            else dep[e[id].v]=0;
        }
    }
    return F;
}
int Dinic()
{
    int flow=0;
    while(bfs())
    {
        for(int i=1;i<=n;i++)cur[i]=0;
        while(int d=dfs(s,Inf))flow+=d;
    }
    return flow;
}
signed main()
{
    n=gi();m=gi();s=gi();t=gi();
    for(int i=0;i<m;i++)
    {
        int u=gi(),v=gi(),c=gi();
        E[i]=(node){u,v,c};
    }
    sort(E,E+m,cmp);
    for(int type=0;type<2;type++){
        for(int p=1<<30,now=0;p;p>>=1)
        {
            while(now<m && E[now].c>=p)
            {
                if(!type)Add(E[now].u,E[now].v,E[now].c);
                else front[E[now].v].push_back(now*2+1);
                now++;
            }
            ans+=Dinic();
        }
    }
    printf("%lld\n",ans);
    return 0;
}

不一样的网络流系列——Dinic跑得快的更多相关文章

  1. [知识点]网络流之Dinic算法

    // 此博文为迁移而来,写于2015年2月6日,不代表本人现在的观点与看法.原始地址:http://blog.sina.com.cn/s/blog_6022c4720102vrg4.html      ...

  2. [无效]网络流之Dinic算法

    // 此博文为迁移而来,写于2015年2月6日,不代表本人现在的观点与看法.原始地址:http://blog.sina.com.cn/s/blog_6022c4720102vrg4.html UPDA ...

  3. 网络流系列算法总结(bzoj 3438 1061)

    网络流嘛,怎么看都是一堆逗逼题嘛,反正遇到还是都做不起嘛.... 网络流的模板非常简单,难点都在于建图,网络流的建图解决问题范围之广,下至A+B Problem,上至单纯形,线性规划.所以如果对于网络 ...

  4. 「CODVES 1922 」骑士共存问题(二分图的最大独立集|网络流)&dinic

    首先是题目链接  http://codevs.cn/problem/1922/ 结果发现题目没图(心情复杂 然后去网上扒了一张图 大概就是这样了. 如果把每个点和它可以攻击的点连一条边,那问题就变成了 ...

  5. 网络流 KM dinic

    study from: https://blog.csdn.net/A_Comme_Amour/article/details/79356220 1. Edmonds-Karp 无优化 最坏时间复杂度 ...

  6. 初涉网络流[EK&dinic]

    主要还是板子 Edmonds-Karp 从S开始bfs,直到找到一条到达T的路径后将该路径增广,并重复这一过程. 在处理过程中,为了应对“找到的一条路径把其他路径堵塞”的情况,采用了建反向弧的方式来实 ...

  7. 网络流之Dinic算法

    初学网络流.存一下Dinic板子. 复杂度O(n^2*m) UVA - 1515 Pool construction 把每个草地与 S 相连,花费为dig,每个洞与 T 相连,花费为 然后对于每个两个 ...

  8. 网络流 之 dinic 算法

    网络流指的是:网络流(network-flows)是一种类比水流的解决问题方法.(类似于水管群,有一个源点(水无限多),和一个汇点,最大流就代表这个点水管群(边集)每秒最大能送道汇点的水量) 这个怎么 ...

  9. 初探网络流:dinic/EK算法学习笔记

    前记 这些是初一暑假的事: "都快初二了,连网络流都不会,你好菜啊!!!" from 某机房大佬 to 蒟蒻我. flag:--NOIP后要学网络流 咕咕咕------------ ...

随机推荐

  1. Java SE学习【三】——JDBC

    最近学到了数据库与java的jdbc方面,还有个DAO模式,写一下自己的理解,后期有什么不对的再改. 一.数据库三范式的理解 记得以前上课时,也上了一学期的“数据库系统原理”,给我们上课的老师算是渣渣 ...

  2. php实现MySQL两库对比升级版

    define('DATABASE1', 'db1'); $dbi1 = new DbMysql; $dbi1->dbh = 'mysql://root:password@127.0.0.1/'. ...

  3. HDU 6129 Just do it

    题意:给你一个包含n个数的序列A和一个数m,序列B中的数是序列A经过异或得到的,比如:b[i]=a[1]^a[2]^…..^a[i].现在让你求经过m次异或后,序列B的值.  思路:这题其实和杨辉三角 ...

  4. tp5中代替tp3.2中的一些方法

    U方法 U方法是TP中的生成路由的内置方法,现在这个方法可以完全使用url方法替换 I方法 之前的TP有个I方法用来接收请求参数,目前可以使用input方法替代 C方法 c方法被config方法代替

  5. vue-cli引入mui的步骤

    不用npm安装了 1.mui官方GitHub下载mui所需文件 https://github.com/dcloudio/mui 把下载来的dist文件夹整个复制到static文件夹中 2.在index ...

  6. CSS Sprites (CSS 精灵) 技术

    CSS Sprites在国内很多人叫css精灵,是一种网页图片应用处理方式.它允许你将一个页面涉及到的所有零星图片都包含到一张大图中去,这样一来,当访问该页面时,载入的图片就不会像以前那样一幅一幅地慢 ...

  7. yml的mybatis的sql查看;Mybatis+Springboot 控制台查看日志,Mybatis结合springboot打印日志

    1.配置如图 文件为yml mybatis: mapper-locations: classpath:com/springboot/transaction/mapper/*.xml configura ...

  8. 利用tcpcopy引流过程

    tcpcopy是一个tcp流量复制工具,当前还支持udp和mysql流量的复制. 目的: 将机器10.24.110.21的5000端口流量引流到机器10.23.25.11的5000端口. 示例:将10 ...

  9. 适配android和iOS上position:absolute和input问题

    //适配android上absolute和input的问题var oHeight = $(document).height(); //屏幕当前的高度$(window).resize(function( ...

  10. uwsgi_read_timeout超时处理

    最近发现一服务器一个奇怪的现象: Django的视图函数在浏览器一个请求的情况下,竟然做了两个请求的函数处理.不可思议,找了几天也不知道为什么, 只发现只要用uwsgi_read_timeout之后, ...