hdu 6437 /// 最小费用最大流 负花费 SPFA模板

题目大意:

给定n,m,K,W

表示n个小时 m场电影(分为类型A、B)

K个人 若某个人连续看了两场相同类型的电影则失去W 电影时间不能重叠

接下来给定m场电影的 s t w op

表示电影的 开始时间s 结束时间t 看完这场电影则获得w 电影类型是op(0为A 1为B)

将一场电影拆成两个点 s t,两点间连线花费为-w容量为1

源点与所有电影的s点连线 花费为0容量为1

所有电影的t点与汇点连线 花费为0容量为1

若两场电影的时间不冲突 那么按时间顺序在之间连边

若类型相同 花费为W容量为1 否则 花费为0容量为1

最后设超级源点与源点连边 花费为0容量为K 表示最多K个人

此时求出的 最小花费最大流 的最小花费 就是 最后最少的失去

最少的失去 就是 最大的获得的相反数

#include <bits/stdc++.h>
using namespace std;
#define LL long long
#define INF 0x3f3f3f3f
#define mem(i,j) memset(i,j,sizeof(i))
#define inc(i,l,r) for(int i=l;i<=r;i++)
#define dec(i,r,l) for(int i=r;i>=l;i--)
const int N=405;
const int mod=1e9+7;

int n,m;
struct NODE { int s,t,w,op; }node[205];
struct Edge
{
    int from,to,cap,flow,cost;
    Edge(int u,int v,int ca,int f,int co):from(u),to(v),cap(ca),flow(f),cost(co){};
};

struct MCMF
{
    int n,m,s,t;
    vector<Edge> edges;
    vector<int> G[N];
    int inq[N];//是否在队列中
    int d[N];//距离
    int p[N];//上一条弧
    int v[N];//可改进量

    void init(int n) //初始化 顶点为 0~n-1
    {
        this->n=n;
        for(int i=0;i<=n;i++) G[i].clear();
        edges.clear();
    }

    void AddEdge(int from,int to,int cap,int cost)//加边
    {
        edges.push_back(Edge(from,to,cap,0,cost));
        edges.push_back(Edge(to,from,0,0,-cost));
        int m=edges.size();
        G[from].push_back(m-2);
        G[to].push_back(m-1);
    }

    bool SPFA(int s,int t,int &flow,int &cost)//寻找最小费用的增广路,使用引用同时修改原flow,cost
    {
        for(int i=0;i<n;i++) d[i]=INF;
        memset(inq,0,sizeof(inq));
        d[s]=0;inq[s]=1;p[s]=0;v[s]=INF;
        queue<int>Q; Q.push(s);
        while(!Q.empty()) {
            int u=Q.front(); Q.pop();
            inq[u]--;
            for(int i=0;i<G[u].size();i++) {
                Edge& e=edges[G[u][i]];
                if(e.cap>e.flow && d[e.to]>d[u]+e.cost) {//满足可增广且可变短
                    d[e.to]=d[u]+e.cost;
                    p[e.to]=G[u][i];
                    v[e.to]=min(v[u],e.cap-e.flow);
                    if(!inq[e.to]) inq[e.to]++,Q.push(e.to);
                }
            }
        }
        if(d[t]==INF) return false;//汇点不可达则退出
        flow+=v[t];
        cost+=d[t]*v[t];
        int u=t;
        while(u!=s) {//更新正向边和反向边
            edges[p[u]].flow+=v[t];
            edges[p[u]^1].flow-=v[t];
            u=edges[p[u]].from;
        }
        return true;
    }

    int MincotMaxflow(int s,int t)
    {
        int flow=0,cost=0;
        while(SPFA(s,t,flow,cost));
        return cost;
    }
}MM;

int main()
{
    int _; scanf("%d",&_);
    while(_--) {
        int n,m,K,W;
        scanf("%d%d%d%d",&n,&m,&K,&W);
        MM.init(2*m+3);
        inc(i,1,m) {
            int s,t,w,op;
            scanf("%d%d%d%d",&s,&t,&w,&op);
            node[i]={s,t,w,op};
            MM.AddEdge(i*2,i*2+1,1,-w);
            MM.AddEdge(1,i*2,1,0);
            MM.AddEdge(i*2+1,m*2+2,1,0);
        }
        inc(i,1,m) inc(j,i+1,m) {
            if(node[i].t<=node[j].s) {
                if(node[i].op==node[j].op)
                    MM.AddEdge(i*2+1,j*2,1,W);
                else MM.AddEdge(i*2+1,j*2,1,0);
            }
            if(node[j].t<=node[i].s) {
                if(node[i].op==node[j].op)
                    MM.AddEdge(j*2+1,i*2,1,W);
                else MM.AddEdge(j*2+1,i*2,1,0);
            }
        }
        MM.AddEdge(0,1,K,0);
        int ans=MM.MincotMaxflow(0,m*2+2);
        printf("%d\n",-ans);
    }

    return 0;
}

原文地址:https://www.cnblogs.com/zquzjx/p/10447816.html

时间: 2024-07-30 01:11:17

hdu 6437 /// 最小费用最大流 负花费 SPFA模板的相关文章

hdu 2516 最小费用最大流

原来这个代码超时 #include<stdio.h> #include<queue> #include<string.h> using namespace std; #define N 200 #define inf 0x3fffffff int cap[N][N]; int fee[N][N]; int s,t,sum,pre[N]; int spfa() { queue<int>q; int dis[N],visit[N],u,i; memset(pre

hdu 1533(最小费用最大流)

Going Home Time Limit: 10000/5000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)Total Submission(s): 4223    Accepted Submission(s): 2178 Problem Description On a grid map there are n little men and n houses. In each unit time, every l

多校第一场:HDU 4862 最小费用最大流

思路:这题主要是建图搞了好久,以前没判断过满流,所以又看了这个知识点,然后才发现自己的最小费用最大流在求满流的时候有bug,正好改了过来. 建图:开始看题解知道这题是最小费用最大流,然后没看解释就做了.然后自己建的图没得求出答案,然后想了好久也没发现哪里错.然后看了官方题解,发现自己的建图和官方差太大了.可能是受昨天做POJ最小费用建图的影响吧.官方的建太符合题目意思了,只能说,我还看了好久建图才理解的. 构造二部图,X部有N*M个节点,源点向X部每个节点连一条边,流量1,费用0:Y部有N*M个

hdu 3667(最小费用最大流+拆边)

Transportation Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total Submission(s): 2670    Accepted Submission(s): 1157 Problem Description There are N cities, and M directed roads connecting them. Now you want to

HDU 1533 最小费用最大流(模板)

http://acm.hdu.edu.cn/showproblem.php?pid=1533 这道题直接用了模板 题意:要构建一个二分图,家对应人,连线的权值就是最短距离,求最小费用 要注意void init(int n) 这个函数一定要写 一开始忘记写这个WA了好几发 还有这个题很容易T掉,赋值建图要简化,一开始构建成网络流那种图一直T #include <stdio.h> #include <string.h> #include <iostream> #includ

最小费用最大流(luogu P3381 【模板】最小费用最大流)

题目链接 题目描述 如题,给出一个网络图,以及其源点和汇点,每条边已知其最大流量和单位流量费用,求出其网络最大流和在最大流情况下的最小费用. 输入输出格式 输入格式: 第一行包含四个正整数N.M.S.T,分别表示点的个数.有向边的个数.源点序号.汇点序号. 接下来M行每行包含四个正整数ui.vi.wi.fi,表示第i条有向边从ui出发,到达vi,边权为wi(即该边最大流量为wi),单位流量的费用为fi. 输出格式: 一行,包含两个整数,依次为最大流量和在最大流量情况下的最小费用. 输入输出样例

【网络流#2】hdu 1533 最小费用最大流模板题

嗯~第一次写费用流题... 这道就是费用流的模板题,找不到更裸的题了 建图:每个m(Man)作为源点,每个H(House)作为汇点,各个源点与汇点分别连一条边,这条边的流量是1(因为每个源点只能走一条边到汇点),费用是 从源点走到汇点的步数,因为有多个源点与汇点,要建一个超级源点与超级汇点,超级源点与各个源点连一条流量为1,费用为0(要避免产生多余的费用)的边 按照这个图跑一发费用流即可 把代码挂上去,用的是前向星写的 1 #include<cstdio> 2 #include<cstr

hdu 4494 Teamwork 最小费用最大流

Teamwork Time Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://acm.hdu.edu.cn/showproblem.php?pid=4494 Description Some locations in city A has been destroyed in the fierce battle. So the government decides to send some workers to repair these location

hdu 1533 Going Home 最小费用最大流

题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1533 On a grid map there are n little men and n houses. In each unit time, every little man can move one unit step, either horizontally, or vertically, to an adjacent point. For each little man, you need