zoukankan      html  css  js  c++  java
  • A

    A - ACM Computer Factory

     POJ - 3436 

    As you know, all the computers used for ACM contests must be identical, so the participants compete on equal terms. That is why all these computers are historically produced at the same factory.

    Every ACM computer consists of P parts. When all these parts are present, the computer is ready and can be shipped to one of the numerous ACM contests.

    Computer manufacturing is fully automated by using N various machines. Each machine removes some parts from a half-finished computer and adds some new parts (removing of parts is sometimes necessary as the parts cannot be added to a computer in arbitrary order). Each machine is described by its performance (measured in computers per hour), input and output specification.

    Input specification describes which parts must be present in a half-finished computer for the machine to be able to operate on it. The specification is a set of P numbers 0, 1 or 2 (one number for each part), where 0 means that corresponding part must not be present, 1 — the part is required, 2 — presence of the part doesn't matter.

    Output specification describes the result of the operation, and is a set of Pnumbers 0 or 1, where 0 means that the part is absent, 1 — the part is present.

    The machines are connected by very fast production lines so that delivery time is negligibly small compared to production time.

    After many years of operation the overall performance of the ACM Computer Factory became insufficient for satisfying the growing contest needs. That is why ACM directorate decided to upgrade the factory.

    As different machines were installed in different time periods, they were often not optimally connected to the existing factory machines. It was noted that the easiest way to upgrade the factory is to rearrange production lines. ACM directorate decided to entrust you with solving this problem.

    Input

    Input file contains integers P N, then N descriptions of the machines. The description of ith machine is represented as by 2 P + 1 integers Qi Si,1 Si,2...Si,PDi,1 Di,2...Di,P, where Qi specifies performance, Si,j — input specification for part jDi,k — output specification for part k.

    Constraints

    1 ≤ P ≤ 10, 1 ≤ ≤ 50, 1 ≤ Qi ≤ 10000

    Output

    Output the maximum possible overall performance, then M — number of connections that must be made, then M descriptions of the connections. Each connection between machines A and B must be described by three positive numbers A B W, where W is the number of computers delivered from A to B per hour.

    If several solutions exist, output any of them.

    Sample Input

    Sample input 1
    3 4
    15  0 0 0  0 1 0
    10  0 0 0  0 1 1
    30  0 1 2  1 1 1
    3   0 2 1  1 1 1
    Sample input 2
    3 5
    5   0 0 0  0 1 0
    100 0 1 0  1 0 1
    3   0 1 0  1 1 0
    1   1 0 1  1 1 0
    300 1 1 2  1 1 1
    Sample input 3
    2 2
    100  0 0  1 0
    200  0 1  1 1

    Sample Output

    Sample output 1
    25 2
    1 3 15
    2 3 10
    Sample output 2
    4 5
    1 3 3
    3 5 3
    1 2 1
    2 4 1
    4 5 1
    Sample output 3
    0 0

    Hint

    Bold texts appearing in the sample sections are informative and do not form part of the actual data.
     
     
     
    首先这个题目是一个最小割+dinic+拆点
    这个题目要注意一点就是要建双向边,这个题目和G - Island Transport 这个题目很像,都是双向边,
    但是有一点不一样,那个题目两个点之间只需要建立两条边,这个要建立三条边,这个这个拆点了。
    除此之外就是一个裸的最小割了。
     
     
    #include <cstdio>
    #include <cstdlib>
    #include <cstring>
    #include <string>
    #include <queue>
    #include <vector>
    #include <algorithm>
    #define inf 0x3f3f3f3f
    using namespace std;
    const int INF = 0x3f3f3f3f;
    const int maxn = 1e5 + 10;
    struct edge
    {
        int u, v, c, f;
        edge(int u, int v, int c, int f) :u(u), v(v), c(c), f(f) {}
    };
    vector<edge>e;
    vector<int>G[maxn];
    int level[maxn];//BFS分层,表示每个点的层数
    int iter[maxn];//当前弧优化
    int m;
    void init(int n)
    {
        for (int i = 0; i <= n; i++)G[i].clear();
        e.clear();
    }
    void add(int u, int v, int c)
    {
        e.push_back(edge(u, v, c, 0));
        e.push_back(edge(v, u, 0, 0));
        m = e.size();
        G[u].push_back(m - 2);
        G[v].push_back(m - 1);
    }
    void BFS(int s)//预处理出level数组
    //直接BFS到每个点
    {
        memset(level, -1, sizeof(level));
        queue<int>q;
        level[s] = 0;
        q.push(s);
        while (!q.empty())
        {
            int u = q.front();
            q.pop();
            for (int v = 0; v < G[u].size(); v++)
            {
                edge& now = e[G[u][v]];
                if (now.c > now.f && level[now.v] < 0)
                {
                    level[now.v] = level[u] + 1;
                    q.push(now.v);
                }
            }
        }
    }
    int dfs(int u, int t, int f)//DFS寻找增广路
    {
        if (u == t)return f;//已经到达源点,返回流量f
        for (int &v = iter[u]; v < G[u].size(); v++)
            //这里用iter数组表示每个点目前的弧,这是为了防止在一次寻找增广路的时候,对一些边多次遍历
            //在每次找增广路的时候,数组要清空
        {
            edge &now = e[G[u][v]];
            if (now.c - now.f > 0 && level[u] < level[now.v])
                //now.c - now.f > 0表示这条路还未满
                //level[u] < level[now.v]表示这条路是最短路,一定到达下一层,这就是Dinic算法的思想
            {
                int d = dfs(now.v, t, min(f, now.c - now.f));
                if (d > 0)
                {
                    now.f += d;//正向边流量加d
                    e[G[u][v] ^ 1].f -= d;
                    //反向边减d,此处在存储边的时候两条反向边可以通过^操作直接找到
                    return d;
                }
            }
        }
        return 0;
    }
    int Maxflow(int s, int t)
    {
        int flow = 0;
        for (;;)
        {
            BFS(s);
            if (level[t] < 0)return flow;//残余网络中到达不了t,增广路不存在
            memset(iter, 0, sizeof(iter));//清空当前弧数组
            int f;//记录增广路的可增加的流量
            while ((f = dfs(s, t, INF)) > 0)
            {
                flow += f;
            }
        }
        return flow;
    }
    
    int main()
    {
        int n, m;
        while(scanf("%d%d",&n,&m)!=EOF)//n是城市的数量,m是高速公路的数量
        {
            init(2*n+10);
            int s, t, x, y;
            scanf("%d%d", &s, &t);
            for(int i=1;i<=n;i++)
            {
                scanf("%d", &x);
                add(i, i + n, x);
            }
            for(int i=1;i<=m;i++)
            {
                scanf("%d%d", &x, &y);
                add(x + n, y, inf);
                add(y+n, x, inf);
            }
            int ans = Maxflow(s, t+n);
            printf("%d
    ", ans);
        }
        return 0;
    }
     
     

     
     
     
  • 相关阅读:
    JavaScript的for循环
    javaScript的执行机制-同步任务-异步任务-微任务-宏任务
    js排他性算法
    js倒计时
    微信小程序 简单获取屏幕视口高度
    小程序scroll-view实现回到顶部
    Vue使用js鼠标蜘蛛特效
    小程序获取当前播放长度和视频总长度,可在播放到某一时长暂停或停止视频
    Django学习笔记
    SQLite简单介绍
  • 原文地址:https://www.cnblogs.com/EchoZQN/p/10820179.html
Copyright © 2011-2022 走看看