P4542 [ZJOI2011]营救皮卡丘(Floyd+网络流)

P4542 [ZJOI2011]营救皮卡丘

乍一看似乎没啥题相似的

仔细一看,$N<=150$

边又是双向边,似乎可以用Floyd搞

先跑一遍Floyd处理出$dis[i][j]$

注意到走据点要先走小的才能走大的

也就是说,$i<j<k$时,$dis[i][j]$不能从$k$转移过来

并且实际走路径时,编号也必须从小到大

于是题目转化成了:

给定序列$0,1,2,3,.....,n-1,n$,给出每两个数字之间的转移代价$dis[i][j](i<j)$,

用$k$条从0开始的子序列覆盖整条序列的最小代价,且每个数(除0外)恰好被覆盖一次。(你闲着没事走两次干啥)

有没有可能$i$到$j$的最优路径中间经过$k$,使$k$被覆盖2次?不会,因为在Floyd中已经处理掉了。

卧槽这不是P2469 [SDOI2010]星际竞速吗!

于是就转化为一个最小路径覆盖问题写辣

#include<iostream>
#include<cstdio>
#include<cstring>
#include<queue>
using namespace std;
#define N 505
#define M 200005
int n,m,k,S,T0,T,tC,dis[N][N],d[N],a[N],p[N];
queue <int> h; bool inh[N];
int cnt=1,hd[N],nxt[M],ed[N],poi[M],val[M],cst[M];
inline void adde(int x,int y,int v1,int v2){
    nxt[ed[x]]=++cnt, hd[x]=hd[x]?hd[x]:cnt,
    ed[x]=cnt, poi[cnt]=y, val[cnt]=v1, cst[cnt]=v2;
}
inline void link(int x,int y,int v1,int v2){adde(x,y,v1,v2),adde(y,x,0,-v2);}
bool bfs(){
    memset(d,63,sizeof(d)); int Inf=d[0];
    h.push(S); inh[S]=1; a[S]=Inf; d[S]=0;
    while(!h.empty()){
        int x=h.front(); h.pop(); inh[x]=0;
        for(int i=hd[x];i;i=nxt[i]){
            int to=poi[i];
            if(val[i]>0&&d[to]>d[x]+cst[i]){
                d[to]=d[x]+cst[i]; p[to]=i;
                a[to]=min(a[x],val[i]);
                if(!inh[to]) h.push(to),inh[to]=1;
            }
        }
    }if(d[T]==Inf) return 0;
    tC+=a[T]*d[T];
    for(int i=T;i!=S;i=poi[p[i]^1])
        val[p[i]]-=a[T],val[p[i]^1]+=a[T];
    return 1;
}
void Floyd(){
    memset(dis,63,sizeof(dis));
    for(int i=0;i<=n;++i) dis[i][i]=0;
    for(int i=1,u,v,w;i<=m;++i){
        scanf("%d%d%d",&u,&v,&w);
        dis[u][v]=dis[v][u]=min(dis[v][u],w);
    }
    for(int k=0;k<=n;++k)
        for(int i=0;i<=n;++i)
            for(int j=0;j<=n;++j)
                if(k<=i||k<=j)//不能从大的转移回来
                    dis[i][j]=min(dis[i][j],dis[i][k]+dis[k][j]);
}
int main(){
    scanf("%d%d%d",&n,&m,&k); Floyd();
    S=n*2+2; T=S+1;
    for(int i=1;i<=n;++i)
        link(S,i,1,0),link(i+n+1,T,1,0);
    link(S,0,k,0);//最大流量限制为k
    for(int i=0;i<=n;++i)
        for(int j=i+1;j<=n;++j)
                link(i,j+n+1,1,dis[i][j]);
    while(bfs());
    printf("%d",tC);
    return 0;
}

原文地址:https://www.cnblogs.com/kafuuchino/p/10805119.html

时间: 2024-10-09 14:39:20

P4542 [ZJOI2011]营救皮卡丘(Floyd+网络流)的相关文章

BZOJ 2324: [ZJOI2011]营救皮卡丘( floyd + 费用流 )

昨晚写的题...补发一下题解... 把1~N每个点拆成xi, yi 2个. 预处理i->j经过编号不超过max(i,j)的最短路(floyd) S->0(K, 0), S->xi(1, 0)(从i点继续走), 0->yi(1, distance(0->i))(从0出发), xi->yi(1, distance(i->j))(i点走向j点), yi->T(1, 0)(每个点必须经过至少一次), 然后跑最小费用最大流, 费用即为答案. 写完这道题感觉...只是会

bzoj 2324 [ZJOI2011]营救皮卡丘(floyd,费用流)

2324: [ZJOI2011]营救皮卡丘 Time Limit: 10 Sec  Memory Limit: 256 MBSubmit: 1777  Solved: 712[Submit][Status][Discuss] Description 皮卡丘被火箭队用邪恶的计谋抢走了!这三个坏家伙还给小智留下了赤果果的挑衅!为了皮卡丘,也为了正义,小智和他的朋友们义不容辞的踏上了营救皮卡丘的道路. 火箭队一共有N个据点,据点之间存在M条双向道路.据点分别从1到N标号.小智一行K人从真新镇出发,营救

【BZOJ2324】[ZJOI2011]营救皮卡丘 有上下界费用流

[BZOJ2324][ZJOI2011]营救皮卡丘 Description 皮卡丘被火箭队用邪恶的计谋抢走了!这三个坏家伙还给小智留下了赤果果的挑衅!为了皮卡丘,也为了正义,小智和他的朋友们义不容辞的踏上了营救皮卡丘的道路. 火箭队一共有N个据点,据点之间存在M条双向道路.据点分别从1到N标号.小智一行K人从真新镇出发,营救被困在N号据点的皮卡丘.为了方便起见,我们将真新镇视为0号据点,一开始K个人都在0号点. 由于火箭队的重重布防,要想摧毁K号据点,必须按照顺序先摧毁1到K-1号据点,并且,如

BZOJ 2324: [ZJOI2011]营救皮卡丘

2324: [ZJOI2011]营救皮卡丘 Time Limit: 10 Sec  Memory Limit: 256 MBSubmit: 2340  Solved: 963[Submit][Status][Discuss] Description 皮卡丘被火箭队用邪恶的计谋抢走了!这三个坏家伙还给小智留下了赤果果的挑衅!为了皮卡丘,也为了正义,小智和他的朋友们义不容辞的踏上了营救皮卡丘的道路. 火箭队一共有N个据点,据点之间存在M条双向道路.据点分别从1到N标号.小智一行K人从真新镇出发,营救

【bzoj2324】[ZJOI2011]营救皮卡丘 最短路-Floyd+有上下界费用流

原文地址:http://www.cnblogs.com/GXZlegend/p/6832504.html 题目描述 皮卡丘被火箭队用邪恶的计谋抢走了!这三个坏家伙还给小智留下了赤果果的挑衅!为了皮卡丘,也为了正义,小智和他的朋友们义不容辞的踏上了营救皮卡丘的道路. 火箭队一共有N个据点,据点之间存在M条双向道路.据点分别从1到N标号.小智一行K人从真新镇出发,营救被困在N号据点的皮卡丘.为了方便起见,我们将真新镇视为0号据点,一开始K个人都在0号点. 由于火箭队的重重布防,要想摧毁K号据点,必须

BZOJ2324 [ZJOI2011]营救皮卡丘 【费用流】

题目 皮卡丘被火箭队用邪恶的计谋抢走了!这三个坏家伙还给小智留下了赤果果的挑衅!为了皮卡丘,也为了正义,小智和他的朋友们义不容辞的踏上了营救皮卡丘的道路. 火箭队一共有N个据点,据点之间存在M条双向道路.据点分别从1到N标号.小智一行K人从真新镇出发,营救被困在N号据点的皮卡丘.为了方便起见,我们将真新镇视为0号据点,一开始K个人都在0号点. 由于火箭队的重重布防,要想摧毁K号据点,必须按照顺序先摧毁1到K-1号据点,并且,如果K-1号据点没有被摧毁,由于防御的连锁性,小智一行任何一个人进入据点

【BZOJ 2324】 [ZJOI2011]营救皮卡丘

Description 皮卡丘被火箭队用邪恶的计谋抢走了!这三个坏家伙还给小智留下了赤果果的挑衅!为了皮卡丘,也为了正义,小智和他的朋友们义不容辞的踏上了营救皮卡丘的道路. 火箭队一共有N个据点,据点之间存在M条双向道路.据点分别从1到N标号.小智一行K人从真新镇出发,营救被困在N号据点的皮卡丘.为了方便起见,我们将真新镇视为0号据点,一开始K个人都在0号点. 由于火箭队的重重布防,要想摧毁K号据点,必须按照顺序先摧毁1到K-1号据点,并且,如果K-1号据点没有被摧毁,由于防御的连锁性,小智一行

bzoj2324[ZJOI2011]营救皮卡丘

题面:一张无向带权连通图,点数n+1(标号0-n),边数m,现在有k个人在0号点,要求依次摧毁1,2,3-.n号点.假如至少一个人经过了点x,就认为点x被摧毁了.只有编号比x小的点都被摧毁才能经过点x. k个人可以分头行动.被摧毁的点在摧毁之后可以经过.满足要求的前提下,求k个人经过的路径长度之和的最小值.(即每个人走过的路径长度加起来,一个人走一条路多次时长度应多次计算). n<=150,m<=20000,1<=k<=10,边权<=10000 分析:每个点都必须经过一次,有

BZOJ2324 ZJOI2011 营救皮卡丘 最短路+费用流

题意:给定一张无向图,有K个人,每一时刻K个人可以同时走(也可以停在一个节点),在到达i之前必须先到达i-1,求从0到N,K个人走的最小距离和(只需一个人到达即可) 题解: 用Floyd跑出任意两个城市i j间的最短路,更新的前提是k<j(要到达城市j必须先到达1->j-1) 将每个城市拆成两个点A B,u v间连费用为w的边,i为任意一个城市,按如下方式建图: 从A向B连流量为INF费用为0的边,表示一个城市可以经过多次 从S向0B连流量为K费用为0的边,表示最初有K个人从0出发 从iB向T