洛谷 P10P1343 地震逃生 改错

Posted ppprseter

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了洛谷 P10P1343 地震逃生 改错相关的知识,希望对你有一定的参考价值。

P1343 地震逃生

题目描述

汶川地震发生时,四川**中学正在上课,一看地震发生,老师们立刻带领x名学生逃跑,整个学校可以抽象地看成一个有向图,图中有\\(n\\)个点,\\(m\\)条边。1号点为教室,\\(n\\)号点为安全地带,每条边都只能容纳一定量的学生,超过楼就要倒塌,由于人数太多,校长决定让同学们分成几批逃生,只有第一批学生全部逃生完毕后,第二批学生才能从1号点出发逃生,现在请你帮校长算算,每批最多能运出多少个学生,\\(x\\)名学生分几批才能运完。

输入输出格式

输入格式:

第一行3个整数\\(n,m,x(x<2^{31},n<=200,m<=2000)\\);以下\\(m\\)行,每行三个整数\\(a,b,c\\)描述一条边,分别代表从\\(a\\)点到\\(b\\)点有一条边,且可容纳\\(c\\)名学生。

输出格式:

两个整数,分别表示每批最多能运出多少个学生,\\(x\\)名学生分几批才能运完。如果无法到达目的地(\\(n\\)号点)则输出“\\(Orz\\) \\(Ni\\) \\(Jinan\\) \\(Saint\\) \\(Cow!\\)


很明显网络流的裸题。

前几天看到对前向星用\\(x\\) \\(nor\\) 1\\(找反边,觉得很方便,遂用一下,用想到很久没打\\)dinic$了,就决定打打(以前都是偷懒打EK的)

不过这样找反边\\(head\\)最开始时得赋\\(-1\\),而且边的边界也是-1

因为
\\(x\\) \\(nor\\) \\(1=x+1\\)\\(x\\)为偶
\\(x\\) \\(nor\\) \\(1=x-1\\)\\(x\\)为奇

得用上0

然后我.....

技术分享图片

我是得多智障才这样,居然样例还对了...

还有一点,最后算答案是\\((ans-1)/\\)最大流\\(+1\\)

我没给那个\\(ans\\)减一下


code:

#include <cstdio>
#include <queue>
#include <cstring>
using namespace std;
const int N=204;
const int M=2004;
const int inf=0x3f3f3f3f;
struct Edge
{
    int w,to,next;
}edge[M*2];
struct node
{
    int cnt,u;
}S[N];
int n,m,X,cnt=-1,head[N];
int top=0;
void push(int cnt0,int u0) {S[++top].cnt=cnt0,S[top].u=u0;}
void pop() {top--;}
int read()
{
    int x=0;char c=getchar();
    while(c<‘0‘||c>‘9‘) c=getchar();
    while(c>=‘0‘&&c<=‘9‘) {x=x*10+c-‘0‘;c=getchar();}
    return x;
}

void add(int u,int v,int w)
{
    edge[++cnt].next=head[u],edge[cnt].to=v,edge[cnt].w=w,head[u]=cnt;
    edge[++cnt].next=head[v],edge[cnt].to=u,edge[cnt].w=0,head[v]=cnt;
}
int dep[N],used[N],ans=0;
queue <int > q;
bool bfs()
{
    memset(dep,0,sizeof(dep));
    while(!q.empty()) q.pop();
    q.push(1);dep[1]=1;
    while(!q.empty()&&q.front()!=n)
    {
        int u=q.front();
        q.pop();
        for(int i=head[u];i!=-1;i=edge[i].next)
        {
            int v=edge[i].to,w=edge[i].w;
            if(!dep[v]&&w)
            {
                dep[v]=dep[u]+1;
                q.push(v);
            }
        }
    }
    return !q.empty();
}
int main()
{
    memset(head,-1,sizeof(head));
    n=read(),m=read(),X=read();
    int u,v,w;
    for(int i=1;i<=m;i++)
    {
        u=read(),v=read(),w=read();
        add(u,v,w);
    }
    while(bfs())
    {
        memset(used,0,sizeof(used));
        top=0;
        push(head[1],1);
        while(top)
        {
            if(S[top].u==n)
            {
                int m_min=inf,id;
                for(int i=2;i<=top;i++)
                    if(edge[S[i].cnt].w<m_min)
                    {
                        m_min=edge[S[i].cnt].w;
                        id=i;
                    }
                ans+=m_min;
                for(int i=2;i<=top;i++)
                {
                    edge[S[i].cnt].w-=m_min;
                    edge[(S[i].cnt)^1].w+=m_min;
                }
                used[n]=0;
                top=max(0,id-1);
            }
            else
            {
                int u=S[top].u;
                for(int i=head[u];i!=-1;i=edge[i].next)
                {
                    int v=edge[i].to,w=edge[i].w;
                    if(!used[v]&&w&&dep[v]==dep[u]+1)
                    {
                        used[v]=1;
                        push(i,v);
                        break;
                    }
                }
                if(S[top].u==u) top--;
            }
        }
    }
    if(ans==0) {printf("Orz Ni Jinan Saint Cow!\\n");return 0;}
    printf("%d %d\\n",ans,(X-1)/ans+1);
    return 0;
}

2018.5.20



以上是关于洛谷 P10P1343 地震逃生 改错的主要内容,如果未能解决你的问题,请参考以下文章

洛谷1343地震逃生

洛谷 P1343 地震逃生

地震逃生

LG1343 地震逃生

luogu 1343 地震逃生

luogu P1343 地震逃生 题解