zoukankan      html  css  js  c++  java
  • [ BZOJ 2127 ]happiness(最小割之二元关系)

    高一一班的座位表是个n*m的矩阵,经过一个学期的相处,每个同学和前后左右相邻的同学互相成为了好朋友。这学期要分文理科了,每个同学对于选择文科与理科有着自己的喜悦值,而一对好朋友如果能同时选文科或者理科,那么他们又将收获一些喜悦值。

    作为计算机竞赛教练的scp大老板,想知道如何分配可以使得全班的喜悦值总和最大。

    输入格式

    第一行两个正整数n,m。

    接下来是六个矩阵

    • 第一个矩阵为n行m列

    此矩阵的第i行第j列的数字表示座位在第i行第j列的同学选择文科获得的喜悦值。

    • 第二个矩阵为n行m列

    此矩阵的第i行第j列的数字表示座位在第i行第j列的同学选择理科获得的喜悦值。

    • 第三个矩阵为n-1行m列

    此矩阵的第i行第j列的数字表示座位在第i行第j列的同学与第i+1行第j列的同学同时选择文科获得的额外喜悦值。

    • 第四个矩阵为n-1行m列

    此矩阵的第i行第j列的数字表示座位在第i行第j列的同学与第i+1行第j列的同学同时选择理科获得的额外喜悦值。

    • 第五个矩阵为n行m-1列

    此矩阵的第i行第j列的数字表示座位在第i行第j列的同学与第i行第j+1列的同学同时选择文科获得的额外喜悦值。

    • 第六个矩阵为n行m-1列

    此矩阵的第i行第j列的数字表示座位在第i行第j列的同学与第i行第j+1列的同学同时选择理科获得的额外喜悦值。

    输出格式

    输出一个整数,表示喜悦值总和的最大值

    输入输出样例

    输入 #1
    1 2
    1 1
    100 110
    1
    1000
    输出 #1
    1210
    

    说明/提示

    【样例说明】

    两人都选理,则获得100+110+1000的喜悦值。

    对于100%以内的数据,n,m<=100 所有喜悦值均为小于等于5000的非负整数

    SOLUTION:

    这篇:https://blog.csdn.net/hzj1054689699/article/details/53038620

    本来每连个点都建立一个二元关系,可后来发现这样直接两个点就被独立出去了,这样是不行的

    CODE:

    #include<iostream>
    #include<cstdio>
    #include<cstdlib>
    #include<cstring>
    #include<cmath>
    #include<algorithm>
    #include<set>
    #include<map>
    #include<vector>
    #include<queue>
    using namespace std;
    #define MAXL 500000
    #define MAX 50000
    #define INF 1000000000
    #define MAXN 120
    inline int read()
    {
        int x=0,t=1;char ch=getchar();
        while((ch<'0'||ch>'9')&&ch!='-')ch=getchar();
        if(ch=='-')t=-1,ch=getchar();
        while(ch<='9'&&ch>='0')x=x*10+ch-48,ch=getchar();
        return x*t;
    }
    struct Line
    {
        int v,next,w;
    }e[MAXL];
    int h[MAX],cnt;
    int S,T,n,m,K;
    inline void Add(int u,int v,int w)
    {
        e[cnt]=(Line){v,h[u],w};h[u]=cnt++;
        e[cnt]=(Line){u,h[v],0};h[v]=cnt++;
    }
    inline void Add2(int u,int v,int w)
    {
        e[cnt]=(Line){v,h[u],w};h[u]=cnt++;
        e[cnt]=(Line){u,h[v],w};h[v]=cnt++;
    
    }
    int level[MAX];
    bool BFS()
    {
        memset(level,0,sizeof(level));
        level[S]=1;
        queue<int> Q;
        Q.push(S);
        while(!Q.empty())
        {
            int u=Q.front();Q.pop();
            for(int i=h[u];i!=-1;i=e[i].next)
            {
                int v=e[i].v;
                if(e[i].w&&!level[v])
                    level[v]=level[u]+1,Q.push(v);
            }
        }
        return level[T];
    }
    int DFS(int u,int flow)
    {
        if(flow==0||u==T)return flow;
        int ret=0;
        for(int i=h[u];i!=-1;i=e[i].next)
        {
            int v=e[i].v;
            if(e[i].w&&level[v]==level[u]+1)
            {
                int dd=DFS(v,min(flow,e[i].w));
                flow-=dd;ret+=dd;
                e[i].w-=dd;e[i^1].w+=dd;
            }
        }
        return ret;
    }
    int Dinic()
    {
        int ret=0;
        while(BFS())ret+=DFS(S,INF);
        return ret;
    }
    int bh[MAXN][MAXN];
    int g[10][MAXN][MAXN];
    int a[MAXN][MAXN];
    int b[MAXN][MAXN];
    int tot,ans;
    int main()
    {
        memset(h,-1,sizeof(h));
        n=read();m=read();
        for(int i=1;i<=n;++i)
            for(int j=1;j<=m;++j)
                bh[i][j]=++tot;
        S=0;T=tot+1;
        //S文科 T理科
        for(int Mx=1;Mx<=6;++Mx)
        {
            int xx=n,yy=m;
            if(Mx==3||Mx==4)xx--;
            if(Mx==5||Mx==6)yy--;
            for(int i=1;i<=xx;++i)
                for(int j=1;j<=yy;++j)
                {
                    ans+=g[Mx][i][j]=read();
                    if(Mx==3)a[i][j]+=g[Mx][i][j],a[i+1][j]+=g[Mx][i][j];
                    if(Mx==4)b[i][j]+=g[Mx][i][j],b[i+1][j]+=g[Mx][i][j];
                    if(Mx==5)a[i][j]+=g[Mx][i][j],a[i][j+1]+=g[Mx][i][j];
                    if(Mx==6)b[i][j]+=g[Mx][i][j],b[i][j+1]+=g[Mx][i][j];
                }
        }
    
    
        for(int i=1;i<=n;++i)
            for(int j=1;j<=m;++j)
            {
                Add(S,bh[i][j],a[i][j]+g[1][i][j]*2);
                Add(bh[i][j],T,b[i][j]+g[2][i][j]*2);
            }
    
        for(int i=1;i<=n;++i)
            for(int j=1;j<=m;++j)
            {
                if(i!=1)Add2(bh[i][j],bh[i-1][j],g[3][i-1][j]+g[4][i-1][j]);
                if(j!=1)Add2(bh[i][j],bh[i][j-1],g[5][i][j-1]+g[6][i][j-1]);
            }
    
        printf("%d
    ",ans-Dinic()/2);
        return 0;
    }
    

      

  • 相关阅读:
    github教程
    Django订单接入支付宝
    python去除html标签的几种方法
    vue-cli项目生成
    restful设计规范
    Vue的指令系统、计算属性和表单输入绑定
    Vue工具
    药物不良反应数据库信息的下载
    爬虫案例之Pubmed数据库下载
    数据分析案例之39药品网
  • 原文地址:https://www.cnblogs.com/zhangbuang/p/11301384.html
Copyright © 2011-2022 走看看