网络流 HDU - 3416 每边只能经过一次的最短路的条数 网络流

题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=3416
题目大意:有n座城市m条单向边。问从A城市到B城市有多少种走法,每条边只能经过一次。
图片说明
图片说明
思路:我们跑出最短路图,建图流量为1,然后直接最短流。

#include<bits/stdc++.h>
#define LL long long
using namespace std;
const LL maxn=100005;
const LL INF=10000000000000000;
LL d[maxn], cur[maxn], start, tend;
struct node
{
    LL to, cap, next;
} edge[maxn << 1];
LL head[maxn];

LL cnt;
void addedge(LL start, LL to, LL cap)
{
    edge[cnt].to = to;
    edge[cnt].cap = cap;
    edge[cnt].next = head[start];
    head[start] = cnt++;
}
bool BFS()
{
    //memset(vis,false,sizeof(vis));
    memset(d, -1, sizeof(LL)*1005);
    LL Q[maxn * 2];
    LL Thead, Ttail;
    Thead = Ttail = 0;
    Q[Ttail++] = start;;
    d[start] = 0;
    //vis[start]=1;
    while (Thead<Ttail)
    {
        LL x = Q[Thead];
        if (x == tend)
            return true;
        for (LL i = head[x]; i != -1; i = edge[i].next)
        {
            LL temp = edge[i].to;
            if (d[temp] == -1 && edge[i].cap>0)//没有标记,且可行流大于0
            {
                //vis[temp.to]=true;
                d[temp] = d[x] + 1;
                Q[Ttail++] = temp;
            }
        }
        Thead++;
    }
    return false;//汇点是否成功标号,也就是说是否找到增广路
}
LL DFS(LL x, LL cap)
{
    if (x == tend)
        return cap;
    //vis[start]=true;
    LL flow = 0, f;
    for (LL i = head[x]; i != -1; i = edge[i].next)
    {
        LL temp = edge[i].to;
        //if(temp.cap<=0||vis[temp.to])continue;
        if (d[temp] == d[x] + 1 && edge[i].cap)
        {
            f = DFS(temp, min(cap - flow, edge[i].cap));
            edge[i].cap -= f;
            edge[i ^ 1].cap += f;
            flow += f;
            if (flow == cap)
                break;
        }
    }
    if (!flow)
        d[x] = -2;//防止重搜
    return flow;
}
LL maxflow()
{
    LL flow = 0, f;
    while (BFS())
    {
        //memset(vis,false,sizeof(vis));
        while ((f = DFS(start, INF))>0)
            flow += f;
    }
    return flow;
}

vector< pair<LL ,LL > > e1[maxn], e2[maxn];
LL n, m;
LL dis[maxn];   //dis当前的最短路
LL vis[maxn];   //是否已经求出最短路
LL dis2[maxn];   //dis当前的最短路
LL vis2[maxn];   //是否已经求出最短路
LL xx[maxn], yy[maxn], zz[maxn];

priority_queue<pair<LL, LL> > q;
LL dijkstra(LL s, LL t)
{
    while(!q.empty())
    {
        q.pop();
    }
    dis[s]=0;
    q.push(make_pair(-dis[s], s));
    while(!q.empty())
    {
        LL now=q.top().second;                //一定是最短路上的顶点
        q.pop();
        if(vis[now])
        {
            continue;
        }
        vis[now]=1;
        for(LL i=0;i<e1[now].size();i++)       //访问所有的邻接点
        {
            LL v=e1[now][i].first;
            if(!vis[v]&&dis[v]>dis[now]+e1[now][i].second)
            {
                dis[v]=dis[now]+e1[now][i].second;
                q.push(make_pair(-dis[v], v));//把可能的最短路顶点加入队列
            }
        }
    }
    if(dis[t]==INF)
    {
        return -1;                            //s-t没有通路
    }
    else
    {
        return dis[t];
    }

}

LL dijkstra2(LL s, LL t)
{
    memset(vis, 0 ,sizeof(vis));
    while(!q.empty())
    {
        q.pop();
    }
    dis2[s]=0;
    q.push(make_pair(-dis2[s], s));
    while(!q.empty())
    {
        LL now=q.top().second;                //一定是最短路上的顶点
        q.pop();
        if(vis[now])
        {
            continue;
        }
        vis[now]=1;
        for(LL i=0;i<e2[now].size();i++)       //访问所有的邻接点
        {
            LL v=e2[now][i].first;
            if(!vis[v]&&dis2[v]>dis2[now]+e2[now][i].second)
            {
                dis2[v]=dis2[now]+e2[now][i].second;
                q.push(make_pair(-dis2[v], v));//把可能的最短路顶点加入队列
            }
        }
    }
    if(dis2[t]==INF)
    {
        return -1;                            //s-t没有通路
    }
    else
    {
        return dis2[t];
    }

}

int main()
{
    LL t;
    scanf("%lld",&t);
    while(t--)
    {
        LL n, m;
        scanf("%lld%lld",&n,&m);
        for(LL i=0;i<maxn;i++)   //初始化
        {
            vis[i]=0;
            dis[i]=INF;
            dis2[i]=INF;
            e1[i].clear();
            e2[i].clear();
        }

        for(LL i=0;i<m;i++)
        {
            LL x, y, z;
            scanf("%lld%lld%lld",&x,&y,&z);
            xx[i]=x, yy[i]=y, zz[i]=z;
            e1[x].push_back(make_pair(y ,z));
            e2[y].push_back(make_pair(x ,z));
        }
        scanf("%lld%lld", &start, &tend);
        dijkstra(start, tend);
        dijkstra2(tend, start);

        cnt = 0;
        memset(head, -1, sizeof(head));


        for(LL i=0; i<=m; i++)
        {
            LL u=xx[i], v=yy[i], w=zz[i];
            if(dis[u]+dis2[v]+w==dis[tend])
            {
                addedge(u, v, 1);
                addedge(v, u, 0);
            }
        }
        cout<<maxflow()<<endl;

    }
    return 0;
}
全部评论

相关推荐

2024-11-05 07:29
贵州大学 Java
点赞 评论 收藏
分享
牛客539033066号:放心吧,这里面一大半都不会去面试的,剩下一半面过了最后还是回拒,实际上免笔试的那些bg的人,没多少愿意去这些岗位,薪资水平在那里
点赞 评论 收藏
分享
评论
点赞
收藏
分享
牛客网
牛客企业服务