1658 - Admiral (拆点+最小费用流)

该题中的拆点法是解决几点容量的通用方法 。  因为只有容量限制的话仍然不能满足每个结点只访问一次这个限制 ,原因很简单,大家画个图就知道了,假设从起点有两条路到同一个结点2,然后又都到末点n,虽然它们满足流量限制但是经过了同一个结点。

那么怎么解决这个问题呢? 答案是:拆点法 。

将一个结点拆成两个结点,由真结点连一条容量为1费用为0的边到假结点,这样之后当我们加边的时候,另起始结点为假结点,终止点为真结点。这样就将这个结点隐性的增加了一个容量属性 。 当然,由于我们要经过起始点和末点两次,所以只能将2~n-1号结点拆点,1和n要特殊处理一下。如果是1或n为边的起点,那么就不要用假结点了。

细节参见代码:

#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
const int INF = 100000000;
const int maxn = 3*1000+5;

int n,m,u,v,c,t,p[maxn],a[maxn],inq[maxn],d[maxn];
struct Edge {
    int from, to, cap, flow, cost;
    Edge(int u,int v,int c,int f,int w):from(u),to(v),cap(c),flow(f),cost(w) {}
};
vector<Edge> edges;
vector<int> g[maxn];
void init() {
    for(int i=0;i<maxn;i++) g[i].clear();
    edges.clear();
}
void AddEdge(int from, int to, int cap, int cost) {
    edges.push_back(Edge(from,to,cap,0,cost));
    edges.push_back(Edge(to,from,0,0,-cost));
    t = edges.size();
    g[from].push_back(t-2);
    g[to].push_back(t-1);
}
bool BellmanFord(int s,int t,int& flow, ll& cost) {
    for(int i=0;i<maxn;i++) d[i] = INF;
    memset(inq,0,sizeof(inq));
    d[s] = 0; inq[s] = 1; p[s] = 0; a[s] = INF;
    queue<int> Q;
    Q.push(s);
    while(!Q.empty()) {
        int u = Q.front(); Q.pop();
        inq[u] = 0;
        for(int i = 0; i < g[u].size(); i++) {
            Edge& e = edges[g[u][i]];
            if(e.cap > e.flow && d[e.to] > d[u] + e.cost) {
                d[e.to] = d[u] + e.cost ;
                p[e.to] = g[u][i];
                a[e.to] = min(a[u],e.cap - e.flow);
                if(!inq[e.to]) { Q.push(e.to); inq[e.to] = 1; }
            }
        }
    }
    if(d[t] == INF) return false;
    flow += a[t];
    cost += (ll)d[t] *(ll)a[t];
    for(int u = t; u != s; u = edges[p[u]].from) {
        edges[p[u]].flow += a[t];
        edges[p[u]^1].flow -= a[t];
    }
    return true;
}
int MincostMaxflow(int s,int t, ll& cost) {
    int flow = 0; cost = 0;
    while(BellmanFord(s,t,flow,cost)) ;
    return flow;
}
int main() {
    while(~scanf("%d%d",&n,&m)&&n) {
        init();
        for(int i=2;i<=n-1;i++) {
            AddEdge(i,i+n,1,0);   //拆点法
        }
        for(int i=1;i<=m;i++) {
            scanf("%d%d%d",&u,&v,&c);//由假结点连向真结点。
            if(u != 1 && u != n) AddEdge(u+n,v,1,c);
            else AddEdge(u,v,1,c);
        }
        ll ans ;
        AddEdge(0,1,2,0);
        AddEdge(n,2*n+1,2,0);
        MincostMaxflow(0,2*n+1,ans);
        printf("%lld\n",ans);//在uva上用I64d会PE
    }
    return 0;
}

版权声明:本文为博主原创文章,未经博主允许不得转载。

时间: 2025-01-07 00:37:14

1658 - Admiral (拆点+最小费用流)的相关文章

UVa 1658,Admiral (拆点+限制最小费用流)

题目链接:https://icpcarchive.ecs.baylor.edu/index.php?option=com_onlinejudge&Itemid=8&page=show_problem&category=569&problem=4277&mosmsg=Submission+received+with+ID+2182664 题目大意:给定一个带权有向图(v,e),求两条互不相交的从1到v的路径(不经过同一个点),使两条路径的权和最小. 分析:即求从1到v

UVa1658 Admiral (拆点法,最小费用流)

链接:http://vjudge.net/problem/UVA-1658 分析:把2到v-1的每个节点i拆成i和i'两个结点,中间连一条容量为1,费用为0的边,然后求1到v的流量为2的最小费用流即可. 1 #include <cstdio> 2 #include<cstring> 3 #include<queue> 4 #include<vector> 5 #include<algorithm> 6 using namespace std; 7

uva 1658 Admiral (最小费最大流)

uva 1658 Admiral 题目大意:在图中找出两条没有交集的线路,要求这两条线路的费用最小. 解题思路:还是拆点建图的问题. 首先每一个点都要拆成两个点.比如a点拆成a->a'.起点和终点的两点间的容量为2费用为0,保证了仅仅找出两条线路.其余点的容量为1费用为0,保证每点仅仅走一遍,两条线路无交集.然后依据题目给出的要求继续建图.每组数据读入a, b, c, 建立a'到b的边容量为1, 费用为c.图建完之后,用bellman-ford来实现MCMF. #include <cstdio

UVa 1658 (拆点法 最小费用流) Admiral

题意: 给出一个有向带权图,求从起点到终点的两条不相交路径使得权值和最小. 分析: 第一次听到“拆点法”这个名词. 把除起点和终点以外的点拆成两个点i和i',然后在这两点之间连一条容量为1,费用为0的边.这样就保证了每个点最多经过一次. 其他有向边的容量也是1 然后求从起点到终点的流量为2(这样就保证了是两条路径)的最小费用流. 本来要在加一个源点和汇点来限制流量的,但是这样弧就多了很多.lrj代码中用了很巧妙的方法,避免了这个问题. 1 #include <bits/stdc++.h> 2

UVa1658 Admiral(拆点法+最小费用流)

题目链接:http://acm.hust.edu.cn/vjudge/problem/viewProblem.action?id=51253 [思路] 固定流量的最小费用流. 拆点,将u拆分成u1和u2,连边(u1,u2,1,0)表示只能经过该点一次.跑流量为2的最小费用流. [代码] 1 #include<cstdio> 2 #include<cstring> 3 #include<queue> 4 #include<vector> 5 #define F

UVa 1658 - Admiral(最小费用最大流 + 拆点)

链接: https://uva.onlinejudge.org/index.php?option=com_onlinejudge&Itemid=8&page=show_problem&problem=4533 题意: 原文地址:https://www.cnblogs.com/hkxy125/p/9539620.html

[题解]uva 1658 Admiral

vjudge传送门[here] 题目大意:给一个有(3≤v≤1000)个点e(3≤e≤10000)条边的有向加权图,求1~v的两条不相交(除了起点和终点外没有公共点)的路径,使权值和最小. 正解是吧2到v-1的每个点拆成两个点,中间连一条容量为1,费用为0的边,然后求1到v的流量为2的最小费用流就行了. Code 1 /** 2 * Uva 3 * Problem#1658 4 * Accepted 5 */ 6 #include<iostream> 7 #include<cstdio&

UVA1658 Admiral 拆点法解决结点容量(路径不能有公共点,容量为1的时候) 最小费用最大流

/** 题目:UVA1658 Admiral 链接:https://vjudge.net/problem/UVA-1658 题意:lrj入门经典P375 求从s到t的两条不相交(除了s和t外,没有公共点)的路径,使得权值和最小. 思路:拆点法. 除了s,t外.把其他点都拆成两个. 例如点A,拆成A和A'.A指向A'连一条容量为1,花费为0的边. 原来指向A的,仍然指向A点. 原来A指向其他点的,由A'指向它们. 最小费用最大流求流量为2时候的最小费用即可. */ #include<iostrea

uva 1658 Admiral 最小费最大流

题意就是让你求两次1到n的最短路.这题应该可以用最短路来求解吧,只需要将第一次用到的边删去即可.我这里是按照算法竞赛入门经典里面提到拆点+最小费最大流. #include<bits/stdc++.h> using namespace std; const int N=1024*4; const int inf=1<<24; struct Edge { int from,to,cap,flow,cost; }; vector<Edge>edges; vector<i