zoukankan      html  css  js  c++  java
  • 【BZOJ-1976】能量魔方Cube 最小割 + 黑白染色

    1976: [BeiJing2010组队]能量魔方 Cube

    Time Limit: 10 Sec  Memory Limit: 64 MB
    Submit: 884  Solved: 307
    [Submit][Status][Discuss]

    Description

    小C 有一个能量魔方,这个魔方可神奇了,只要按照特定方式,放入不同的 能量水晶,就可以产生巨大的能量。 能量魔方是一个 N*N*N 的立方体,一共用 N3 个空格可以填充能量水晶。 能量水晶有两种: ·一种是正能量水晶(Positive) ·一种是负能量水晶(Negative) 当这个魔方被填满后,就会依据填充的能量水晶间的关系产生巨大能量。对 于相邻两(相邻就是拥有同一个面)的两个格子,如果这两个格子填充的是一正一 负两种水晶,就会产生一单位的能量。而整个魔方的总能量,就是这些产生的能 量的总和。 现在,小 C 已经在魔方中填充了一些水晶,还有一些位置空着。他想知道, 如果剩下的空格可以随意填充,那么在最优情况下,这个魔方可以产生多少能量。

    Input

    第一行包含一个数N,表示魔方的大小。 接下来 N2 行,每行N个字符,每个字符有三种可能: P:表示此方格已经填充了正能量水晶; N:表示此方格已经填充了负能量水晶; ?:表示此方格待填充。 上述 N*N 行,第(i-1)*N+1~i*N 行描述了立方体第 i 层从前到后,从左到右的 状态。且每 N 行间,都有一空行分隔。

    Output

    仅包含一行一个数,表示魔方最多能产生的能量

    Sample Input

    2
    P?
    ??

    ??
    N?

    Sample Output

    9

    HINT

    如下状态时,可产生最多的能量。 
    PN 
    NP 

    NP 
    NN 

    【数据规模】 
    10% 的数据N≤3; 
    30% 的数据N≤4; 
    80% 的数据N≤10; 
    100% 的数据N≤40。 

    Source

    Solution

    最小割

    对水晶块黑白染色
    •相邻的水晶之间连容量为1的边
    •对于黑色,与 S 联通表示正能量,与 P 联通表示负能量,对于白色则相反
    •确定的水晶向 S 或 T 连inf边
    •同样最后将总和减去最小割即可

    Code

    #include<iostream>
    #include<cstring>
    #include<algorithm>
    #include<cmath>
    #include<cstdio>
    #include<queue>
    using namespace std;
    int read()
    {
        int x=0,f=1; char ch=getchar();
        while (ch<'0' || ch>'9') {if (ch=='-') f=-1; ch=getchar();}
        while (ch>='0' && ch<='9') {x=x*10+ch-'0'; ch=getchar();}
        return x*f; 
    }
    int N;
    #define MAXM 2000100
    #define MAXN 100
    #define INF 0x7fffffff
    struct EdgeNode{int next,to,cap;}edge[MAXM];
    int head[MAXN*MAXN*MAXN],cnt=1;
    void add(int u,int v,int w) {cnt++; edge[cnt].next=head[u]; head[u]=cnt; edge[cnt].to=v; edge[cnt].cap=w;}
    void insert(int u,int v,int w) {add(u,v,w); add(v,u,0);} 
    int h[MAXN*MAXN*MAXN],cur[MAXN*MAXN*MAXN],S,T;
    bool bfs()
    {
        queue<int>q;
        for (int i=S; i<=T; i++) h[i]=-1;
        h[S]=1; q.push(S);
        while (!q.empty())
            {
                int now=q.front(); q.pop();
                for (int i=head[now]; i; i=edge[i].next)
                    if (h[edge[i].to]==-1 && edge[i].cap)
                        h[edge[i].to]=h[now]+1,q.push(edge[i].to);
            }
        return h[T]!=-1;        
    }
    int dfs(int loc,int low)
    {
        if (loc==T) return low;
        int used=0,w;
        for (int i=cur[loc]; i; i=edge[i].next)
            if (edge[i].cap && h[edge[i].to]==h[loc]+1)
                {
                    w=dfs(edge[i].to,min(edge[i].cap,low-used));
                    edge[i].cap-=w; edge[i^1].cap+=w; used+=w;
                    if (used==low) return low;
                    if (edge[i].to) cur[loc]=i;
                }
        if (!used) h[loc]=-1;
        return used;
    }
    int Dinic()
    {
        int tmp=0;
        while (bfs())
            {
                for (int i=S; i<=T; i++) cur[i]=head[i];
                tmp+=dfs(S,INF);
            }
        return tmp;
    }
    int id[MAXN][MAXN][MAXN],tot,ans;
    char cube[MAXN][MAXN][MAXN];
    void BuildGraph()
    {
        S=0,T=N*N*N+1;
        int ID=0;
        for (int i=1; i<=N; i++)
            for (int j=1; j<=N; j++)
                for (int k=1; k<=N; k++)
                    id[i][j][k]=++ID;
        for (int i=1; i<=N; i++)
            for (int j=1; j<=N; j++)
                for (int k=1; k<=N; k++)
                    {
                        if (i<N) insert(id[i][j][k],id[i+1][j][k],1),insert(id[i+1][j][k],id[i][j][k],1);
                        if (j<N) insert(id[i][j][k],id[i][j+1][k],1),insert(id[i][j+1][k],id[i][j][k],1);
                        if (k<N) insert(id[i][j][k],id[i][j][k+1],1),insert(id[i][j][k+1],id[i][j][k],1);
                    }
        for (int i=1; i<=N; i++)
            for (int j=1; j<=N; j++)
                for (int k=1; k<=N; k++)
                    {
                        if ((i+j+k)%2) 
                            {
                                if (cube[i][j][k]=='P') insert(S,id[i][j][k],INF);
                                if (cube[i][j][k]=='N') insert(id[i][j][k],T,INF);
                            }
                        else 
                            {
                                if (cube[i][j][k]=='P') insert(id[i][j][k],T,INF);
                                if (cube[i][j][k]=='N') insert(S,id[i][j][k],INF);
                            }
                    }
        tot=3*N*N*(N-1);
    }
    int main()
    {
        //ios::sync_with_stdio(false);
        N=read();
        char p;
        for (int i=1; i<=N; i++)
            {
                for (int j=1; j<=N; j++)
                    for (int k=1; k<=N; k++)
                        {
                            cin>>p;
                            if (p=='P') cube[i][j][k]='P';
                            if (p=='?') cube[i][j][k]='?';
                            if (p=='N') cube[i][j][k]='N';
                        }
            }
        BuildGraph();
        ans=Dinic();
        printf("%d
    ",tot-ans);
        return 0;
    }

    用了BeiYu的黑科技,流同步什么的..因为OJ的原因WA了两次QAQ

  • 相关阅读:
    TP框架实现分页及条件查询
    tp框架连贯操作
    php查询
    php修改数据
    php增加数据处理
    php删除数据
    php怎么访问数据库
    php查询
    克隆及加载类
    php静态成员和接口
  • 原文地址:https://www.cnblogs.com/DaD3zZ-Beyonder/p/5679319.html
Copyright © 2011-2022 走看看