码迷,mamicode.com
首页 > 其他好文 > 详细

hdu 3416 Marriage Match IV

时间:2017-09-29 14:59:20      阅读:167      评论:0      收藏:0      [点我收藏+]

标签:math   clear   star   容量   mat   math.h   最大   string   include   

链接:http://acm.hdu.edu.cn/showproblem.php?pid=3416

题意:starvae在城市A,要去城市B约妹子,要走最短路,并且每段路只能走一次,不能重复走。

题解:在走最短路的情况下,有多少条路从A到B,并且每条路是不重复的(路上经过的任何一条边都不可以重复)。首先是求最短路,然后根据所求的最短路找到这写最短路上的边,将这些边建图,求一次最大流。因为边只能走一次,所以每条边的容量是1,不用拆点(因为城市可以多次经过)。

代码:

#include <stdio.h>
#include <algorithm>
#include <math.h>
#include <string.h>
#include <vector>
#include <queue>
#include <map>
#include <stack>
#include <iostream>
#define pi acos(-1.0)
#define INF 0x3f3f3f3f
using namespace std;
#define ll long long
const int maxn=10100;
int head[maxn],cnt=0;
struct edge
{
    int t,f,next;
}e[500010];
void init()
{
    memset(head,-1,sizeof head);
    cnt=0;
}
void add_edge(int u,int v,int f)
{
    e[cnt].t=v;e[cnt].f=f;
    e[cnt].next=head[u];head[u]=cnt++;

    e[cnt].t=u;e[cnt].f=0;
    e[cnt].next=head[v];head[v]=cnt++;
}
vector<pair<int,int> > mp[maxn];
bool vis[maxn];int dis[maxn];
void spfa(int s,int e)
{
    memset(vis,0,sizeof vis);
    memset(dis,INF,sizeof dis);
    queue<int>q;
    q.push(s),dis[s]=0,vis[s]=false;
    while(!q.empty())
    {
        int k=q.front();q.pop();
        vis[k]=0;
        for(int i=0;i<mp[k].size();i++)
        {
            int v=mp[k][i].first,w=mp[k][i].second;
            if(dis[v]>dis[k]+w)
            {
                dis[v]=dis[k]+w;
                if(!vis[v])
                    vis[v]=1,q.push(v);
            }
        }
    }
}
void getedge(int n)
{
    for(int i=1;i<=n;i++)
        for(int j=0;j<mp[i].size();j++)
        {
            int v=mp[i][j].first,w=mp[i][j].second;
            if(dis[v]==dis[i]+w)
            add_edge(i,v,1);//,printf("build %d %d %d\n",i,v,1);
        }
}
bool bfs(int s,int t)
{
    memset(dis,-1,sizeof dis);
    queue<int > q;
    dis[s]=1;q.push(s);
    while(!q.empty())
    {
        int k=q.front();q.pop();
        for(int i=head[k];i!=-1;i=e[i].next)
        {
            int v=e[i].t;
            if(e[i].f>0&&dis[v]==-1)
            {
                dis[v]=dis[k]+1;
                if(v==t) return 1;
                q.push(v);
            }
        }
    }
    return dis[t]!=-1;
}
int cur[maxn];
int dfs(int top,int t,int flow)
{
    if(top==t)
        return flow;
    int ans=0,x=0;
    for(int i=cur[top];~i;i=e[i].next)
    {
        int to=e[i].t;
        if(e[i].f>0&&dis[to]==dis[top]+1)
        {
            x=dfs(to,t,min(flow-ans,e[i].f));
            e[i].f-=x;
            e[i^1].f+=x;
            if(e[i].f)
                cur[top] = i;
            ans+=x;
            if(ans==flow)
                return flow;
        }
    }
    if(ans==0)
        dis[top]=0;
    return ans;
}
int  main()
{
    //freopen("C:\\Users\\Administrator\\Desktop\\a.txt","r",stdin);
    //ios::sync_with_stdio(false);
    //freopen("C:\\Users\\Administrator\\Desktop\\b.txt","w",stdout);
    int T,n,m;
    scanf("%d",&T);
    while(T--)
    {
        scanf("%d%d",&n,&m);
        for(int i=0;i<=n;i++) mp[i].clear();
        init();
        for(int i=0;i<m;i++)
        {
            int a,b,c;
            scanf("%d%d%d",&a,&b,&c);
            mp[a].push_back(make_pair(b,c));
        }
        int s,t;
        scanf("%d%d",&s,&t);
        spfa(s,t);
        getedge(n);
        //printf("\n");
        int ans=0;
        while(bfs(s,t))
        {
            for(int i=0;i<=3*n;i++)
                cur[i]=head[i];
            ans+=dfs(s,t,INF);
        }
        printf("%d\n",ans);
    }
    return 0;
}

 

hdu 3416 Marriage Match IV

标签:math   clear   star   容量   mat   math.h   最大   string   include   

原文地址:http://www.cnblogs.com/MeowMeowMeow/p/7610683.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!