正题

题目链接:https://www.luogu.com.cn/problem/P4542


题目大意

给出\(n+1\)个点\(m\)条边的无向图,\(k\)个人开始在\(0\)号点,一个人进入\(i\)号点之前必须要有人经过\(i-1\)号点,求第一个人进入\(n\)号点时所有人的最短移动距离和。

\(1\leq n\leq 150,1\leq m\leq 2\times 10^4,1\leq k\leq 10\)


解题思路

显然不能建\(n\times n\)个点跑费用流,考虑怎么优化。

我们可以缩去一些中间路程,对于每个人只留下第一次到达该点的这些点,但是我们需要适当改变边权。

用\(Floyd\)求出\(d_{i,j}\)表示从\(i\)走到\(j\)且只走编号不大于\(max\{i,j\}\)的点的最短距离,这样因为如果一个人要走到\(j\),那么它一定是第一个到的,所以不能走过大于\(j\)的点,而前面的我们可以调整每个人的行走顺序来让前面的点都解锁后这个人再出发。

现在问题就变为了求\(k\)条权值和最小的路径覆盖所有点。其实不用上下界,因为是费用流,所以我们每个点拆成出/入点,然后入点向出点连一条\((1,-inf)\)和\((inf,0)\)的边(前面是流量,后面是费用)

这样如果一个点不走会多一堆费用,所以肯定会经过所有点。

这样点数就是\(O(n)\)级别了


code

#include<cstdio>
#include<cstring>
#include<algorithm>
#include<queue>
#define ll long long
using namespace std;
const ll N=310,inf=1e9;
struct node{
ll to,next,w,c;
}a[N*N*10];
ll n,m,k,s,t,ans,tot=1;
ll ls[N],f[N],mf[N],d[N][N],pre[N];
bool v[N];queue<int> q;
void addl(ll x,ll y,ll w,ll c){
a[++tot].to=y;a[tot].next=ls[x];ls[x]=tot;a[tot].w=w;a[tot].c=c;
a[++tot].to=x;a[tot].next=ls[y];ls[y]=tot;a[tot].w=0;a[tot].c=-c;
return;
}
bool spfa(){
memset(f,0x3f,sizeof(f));
f[s]=0;q.push(s);v[s]=1;mf[s]=inf;
while(!q.empty()){
ll x=q.front();q.pop();v[x]=0;
for(ll i=ls[x];i;i=a[i].next){
ll y=a[i].to;
if(a[i].w&&f[x]+a[i].c<f[y]){
f[y]=f[x]+a[i].c;pre[y]=i;
mf[y]=min(mf[x],a[i].w);
if(!v[y])q.push(y),v[y]=1;
}
}
}
return f[t]<1e18;
}
void updata(){
ll x=t;ans+=mf[x]*f[x];
while(x!=s){
a[pre[x]].w-=mf[t];
a[pre[x]^1].w+=mf[t];
x=a[pre[x]^1].to;
}
return;
}
signed main()
{
scanf("%lld%lld%lld",&n,&m,&k);
memset(d,0x3f,sizeof(d));n++;
for(ll i=1;i<=m;i++){
ll x,y,w;
scanf("%lld%lld%lld",&x,&y,&w);x++;y++;
d[x][y]=min(d[x][y],w);
d[y][x]=min(d[y][x],w);
}
for(ll i=1;i<=n;i++)d[i][i]=0;
for(ll k=1;k<=n;k++)
for(ll i=1;i<=n;i++)
for(ll j=1;j<=n;j++)
if(k<i||k<j)d[i][j]=min(d[i][j],d[i][k]+d[k][j]);
s=2*n+1;t=s+1;
addl(s,1,k,0);
for(ll i=1;i<=n;i++){
addl(i,i+n,1,-inf);
addl(i,i+n,inf,0);
addl(i+n,t,inf,0);
for(ll j=i+1;j<=n;j++)
if(d[i][j]<1e18)addl(i+n,j,inf,d[i][j]);
}
while(spfa())
updata();
printf("%lld\n",ans+n*inf);
return 0;
}

P4542-[ZJOI2011]营救皮卡丘【费用流,Floyd】的更多相关文章

  1. BZOJ.2324.[ZJOI2011]营救皮卡丘(费用流 Floyd)

    BZOJ 洛谷 首先预处理出\(dis[i][j]\),表示从\(i\)到\(j\)的最短路.可以用\(Floyd\)处理. 注意\(i,j\)是没有大小关系限制的(\(i>j\)的\(dis[ ...

  2. bzoj2324 [ZJOI2011]营救皮卡丘 费用流

    [ZJOI2011]营救皮卡丘 Time Limit: 10 Sec  Memory Limit: 256 MBSubmit: 2653  Solved: 1101[Submit][Status][D ...

  3. 【BZOJ 2324】[ZJOI2011]营救皮卡丘 费用流

    本人实行诱骗拐卖(利用自然分层与实际意义),正解拼接补充(充分利用最大流限制(不浪费任何一个走出去的机会而不是不浪费任何一个已有的流)与问题转换) #include <cstdio> #i ...

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

    P4542 [ZJOI2011]营救皮卡丘 乍一看似乎没啥题相似的 仔细一看,$N<=150$ 边又是双向边,似乎可以用Floyd搞   先跑一遍Floyd处理出$dis[i][j]$ 注意到走 ...

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

    原文地址:http://www.cnblogs.com/GXZlegend/p/6832504.html 题目描述 皮卡丘被火箭队用邪恶的计谋抢走了!这三个坏家伙还给小智留下了赤果果的挑衅!为了皮卡丘 ...

  6. bzoj 2324 ZJOI 营救皮卡丘 费用流

    题的大概意思就是给定一个无向图,边有权值,现在你有k个人在0点,要求走到n点,且满足 1:人们可以分头行动,可以停在某一点不走了 2:当你走到x时,前x-1个点必须全部走过(不同的人走过也行,即分两路 ...

  7. 洛咕P4542 [ZJOI2011]营救皮卡丘

    套路题? 感觉讲不清,先写建图 把每个点拆成两个,A和B, S->Ai流量=1费用=0,Bi->T流量=1费用=0, Ai->Bj流量=1费用=ij最短路 还有一个特殊的s点,S-& ...

  8. P4542 [ZJOI2011]营救皮卡丘

    题目链接 题意分析 我们仔细分析一下 发现题目要求用最多\(k\)条路径实现最小权覆盖 首先由于最小路径覆盖针对的是有向图 但是这是一个无向图 所以我们面向对象编程 我们维护一个数组\(d[i][j] ...

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

    2324: [ZJOI2011]营救皮卡丘 Time Limit: 10 Sec  Memory Limit: 256 MBSubmit: 1777  Solved: 712[Submit][Stat ...

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

    昨晚写的题...补发一下题解... 把1~N每个点拆成xi, yi 2个. 预处理i->j经过编号不超过max(i,j)的最短路(floyd) S->0(K, 0), S->xi(1 ...

随机推荐

  1. Charles 抓包 Client SSL handshake failed - Remote host closed connection during handshake

    Charles 抓包 https 报错: Client SSL handshake failed - Remote host closed connection during handshake # ...

  2. uniapp封装小程序雷达图组件实现

    效果图: view <canvas id="radar-canvas" class="radar-canvas" type="2d"& ...

  3. java线程池 面试题(精简)

    什么是线程池? 线程池是一种多线程处理形式,处理过程中将任务提交到线程池,任务的执行交由线程池来管理. 如果每个请求都创建一个线程去处理,那么服务器的资源很快就会被耗尽,使用线程池可以减少创建和销毁线 ...

  4. MVVMLight学习笔记(四)---RelayCommand初探

    一.概述 在MVVM Light框架中,主要通过命令绑定来进行事件的处理. WPF中,命令是通过实现 ICommand 接口创建的. ICommand 公开了两个方法(Execute 及 CanExe ...

  5. 十九:JDBC操作事务

    二.MySQL数据库中操作事务命令 2.1.开启事务(start transaction) 使用"start transaction"开启MySQL数据库的事务,如下所示:

  6. ubuntu18.04 开机启动/停止服务

    ubuntu18.04 开机启动/停止服务 一.删除一个服务 如果要删除一个服务,使用uodate-rc.d(参数-f是强制删除符号链接) update-rc.d -f apache2 remove ...

  7. Java异常与异常处理

    异常体系结构 1.所有异常都继承于Throwable类,其下有两大子类: (1)Error类:错误,一般编程人员不太接触,如虚拟机错误.线程死锁.硬伤:使程序崩溃 (2)Exception类:异常,编 ...

  8. mysql基础操作(三):数据约束

    首先创建一个数据库 create database homework default character set utf8; use homework; 1.1 默认值约束(default) -- 数 ...

  9. 【JavasScript】折腾一个基础到不能再基础的顺滑抽奖页面

    前言 事情是这样的,作为一个意志力极低的人,最近一直在找寻提高意志力的方法. 然后决定试一试所谓的"建立奖励机制",也就是说,完成一项意志力挑战后给自己一些奖励(具体操作方法不在这 ...

  10. ORB_SLAM2 闭环检测段错误

    问题描述: Ubuntu14.04运行正常.Ubuntu 16.04下运行时,检测到闭环后有时会段错误,定位发现断错误出现在CorrectLoop()的红色代码处 void LoopClosing:: ...