POJ 3680 区间带权K覆盖 费用流

题目链接:http://poj.org/problem?id=3680
题目大意:有N个带权开区间,让你从中选择一些使得权值最大,实数轴上任意一点被覆盖不超过K次,输出最大权值和。

思路: 经典的在区间上建图题目。首先把所有端点离散化编号从1到n,建立源点0,汇点n+1。对于除汇点之外的点建立从i到i+1的边,容量为K,费用为0。对于每个区间(l,r),建立从l′到r′的边,容量为1,费用为−wi。跑一遍最小费用最大流,结果的相反数即为答案。

#include  <bits/stdc++.h>
#define LL long long
using namespace std;
const LL maxn=500;
const LL maxm=1e6+10;
struct Mcmf_flow{
    bool vis[maxn];
    LL dis[maxn];
    LL pre[maxn];
    LL last[maxn];
    LL flow[maxn];
    LL zdl, fy;
    LL s1[maxn];//sum的前缀和
    LL sum[maxn];//流量为i的最小费用
    LL tot;
    //dis最小花费;pre每个点的前驱;last每个点的所连的前一条边;flow源点到此处的流量
    struct Edge {
        LL to,next,flow,dis;//flow流量 dis花费
    } e[maxm<<1];
    LL head[maxn],cut;
    queue <LL> q;
    int N;
    void init(int n){
        N=n+5;
        memset(s1, 0, sizeof(LL)*(N+60));
        memset(sum, 0, sizeof(LL)*(N+60));
        memset(head,-1,sizeof(LL)*(N+60));
        tot=0;
        cut=-1;//初始化
        zdl=fy=0;
    }

    void add_e(LL from,LL to,LL flow,LL dis) {
        e[++cut].next=head[from];
        e[cut].to=to;
        e[cut].flow=flow;
        e[cut].dis=dis;
        head[from]=cut;
    }

    void add(LL x, LL y, LL z, LL f){
        add_e(x,y,z,f);
        add_e(y,x,0,-f);
    }

    bool spfa(LL s,LL t) {
        memset(dis,0x7f,sizeof(LL)*(N+60));
        memset(flow,0x7f,sizeof(LL)*(N+60));
        memset(vis,0,sizeof(bool)*(N+60));
        q.push(s);
        vis[s]=1; dis[s]=0; pre[t]=-1;
        while (!q.empty()) {
            LL now=q.front();
            q.pop(); vis[now]=0;
            for (LL i=head[now]; i!=-1; i=e[i].next) {
                if (e[i].flow>0 && dis[e[i].to]>dis[now]+e[i].dis) { //正边
                    dis[e[i].to]=dis[now]+e[i].dis;
                    pre[e[i].to]=now;
                    last[e[i].to]=i;
                    flow[e[i].to]=min(flow[now],e[i].flow);//
                    if (!vis[e[i].to]) {
                        vis[e[i].to]=1;
                        q.push(e[i].to);
                    }
                }
            }
        }
        return pre[t]!=-1;
    }

    void MCMF(LL s, LL t) {
        while (spfa(s,t)) {
            LL now=t;
            zdl+=flow[t];
            fy+=flow[t]*dis[t];
            sum[++tot]=fy;//得到sum[]
            s1[tot]=sum[tot]-sum[tot-1];//得到s1[]

            while (now!=s) {
                //从源点一直回溯到汇点
                e[last[now]].flow-=flow[t];//flow和dis容易搞混
                e[last[now]^1].flow+=flow[t];
                now=pre[now];
            }
        }
    }

}min_Flow;


int L[205], R[205], w[205], a[500], tot=0;
int main() {

    int t; scanf("%d", &t);
    while(t--){
        tot=0;
        int n, k; scanf("%d%d", &n, &k);
        for(int i=1; i<=n; i++){
            scanf("%d%d%d", &L[i], &R[i], &w[i]);
            a[++tot]=L[i];
            a[++tot]=R[i];
        }
        sort(a+1, a+tot+1);
        int cnt=unique(a+1, a+tot+1)-a-1;
        for(int i=1; i<=n; i++){
            L[i]=lower_bound(a+1, a+cnt+1, L[i])-a;
            R[i]=lower_bound(a+1, a+cnt+1, R[i])-a;
        }
        int S=0, T=cnt+1;
        min_Flow.init(T);
        for(int i=S; i<T; i++){
            min_Flow.add(i, i+1, k, 0);
        }
        for(int i=1; i<=n; i++){
            min_Flow.add(L[i], R[i], 1, -w[i]);
        }
        min_Flow.MCMF(S, T);
        printf("%d\n", -min_Flow.fy);
    }

    return 0;
}
全部评论

相关推荐

联洲 嵌入式软件开发 总包48w(sp+3档)
点赞 评论 收藏
分享
2024-12-21 01:36
电子科技大学 Java
牛客850385388号:员工福利查看图片
点赞 评论 收藏
分享
评论
点赞
收藏
分享
牛客网
牛客企业服务