P4542 [ZJOI2011]营救皮卡丘
题目链接
题意分析
我们仔细分析一下 发现题目要求用最多\(k\)条路径实现最小权覆盖
首先由于最小路径覆盖针对的是有向图 但是这是一个无向图
所以我们面向对象编程
我们维护一个数组\(d[i][j]\) 表示\(i,j\)之间的最短距离
由于是\(n≤150\) 所以我们可以使用\(floyed\)求
同时由于我们用\(k\)更新\(i,j\)的话 必须是满足 \(k\)小于等于\(max(i,j)\)
因为我们假设已经炸毁了\(u\)
然后去摧毁\(v\) 那么如果我们经过了\(max(u,v)\) 那么\(v\)肯定已经摧毁了
那么再摧毁就没有意义了
然后维护成了一个\(DAG\)
然后我们在\(DAG\)上跑最多\(k\)条最小路径覆盖
我们对于每一个点拆成两个点\(a,b\)
然后源点向\(0_a\)连一条容量为\(k\) 费用为\(0\)的边
源点向\((1-n)_a\)连一条容量为\(1\)费用为\(0\)的边
同时\((0-n)_b\)分别向汇点连一条容量为\(1\)费用为\(0\)的边
然后对于一条边\((x,y)\)(这里强制\(x<y\))
我们由\(x_a\)向\(y_b\)连一条容量为\(1\)费用为\(dis[x][y]\)的边
然后跑最小费用最大流即可
\(DAG\)上的最小不相交路径覆盖问题
我们对于每一个点拆成两个点
然后按照连接关系由于左边的点向右边的点连边
然后\(ans\)=原图顶点数-最大匹配数
\(DAG\)上的最小可相交路径覆盖问题
我们跑一遍\(floyed\)传递闭包
然后做法同上
CODE:
#include<iostream>
#include<cstdio>
#include<cstring>
#include<cmath>
#include<algorithm>
#include<cstdlib>
#include<string>
#include<queue>
#include<map>
#include<stack>
#include<list>
#include<set>
#include<deque>
#include<vector>
#include<ctime>
#define ll long long
#define inf 0x7fffffff
#define N 208
#define IL inline
#define M 1008611
#define D double
#define ull unsigned long long
#define R register
using namespace std;
template<typename T>IL void read(T &_)
{
T __=0,___=1;char ____=getchar();
while(!isdigit(____)) {if(____=='-') ___=0;____=getchar();}
while(isdigit(____)) {__=(__<<1)+(__<<3)+____-'0';____=getchar();}
_=___ ? __:-__;
}
/*-------------OI使我快乐-------------*/
int n,m,k,tot=1,S,T,have;ll ans;
int dis[N][N];
int to[M],nex[M],head[N<<1],w[M],cost[M];
int dep[N<<1],flow[N<<1],pre[N<<1],las[N<<1];
bool vis[N<<1];queue<int> Q;
IL void add(int x,int y,int z,int f)
{to[++tot]=y;nex[tot]=head[x];head[x]=tot;w[tot]=z;cost[tot]=f;
swap(x,y);to[++tot]=y;nex[tot]=head[x];head[x]=tot;w[tot]=0;cost[tot]=-f;}
IL bool spfa()
{
memset(dep,0x3f,sizeof dep);
memset(flow,0x3f,sizeof flow);
memset(vis,0,sizeof vis);
dep[S]=0;vis[S]=1;Q.push(S);pre[T]=-1;
while(!Q.empty())
{
int u=Q.front();Q.pop();vis[u]=0;
for(R int i=head[u];i;i=nex[i])
{
int v=to[i];
if(w[i]>0&&dep[v]>dep[u]+cost[i])
{
dep[v]=dep[u]+cost[i];
flow[v]=min(flow[u],w[i]);
pre[v]=u;las[v]=i;
if(!vis[v]) {vis[v]=1;Q.push(v);}
}
}
}
return pre[T]!=-1;
}
IL void EK()
{
while(spfa())
{
have+=flow[T];
ans+=1ll*flow[T]*dep[T];
for(R int now=T;now!=S;now=pre[now])
{
w[las[now]]-=flow[T];w[las[now]^1]+=flow[T];
}
}
}
int main()
{
// freopen(".in","r",stdin);
// freopen(".out","w",stdout);
read(n);read(m);read(k);++n;S=(n<<1)+1;T=(n<<1)+2;
for(R int i=1;i<=n;++i) dis[i][i]=0;
memset(dis,0x3f,sizeof dis);
for(R int i=1,x,y,z;i<=m;++i)
{
read(x);read(y);read(z);++x;++y;
dis[x][y]=dis[y][x]=min(dis[x][y],z);
}
for(R int p=1;p<=n;++p)
for(R int i=1;i<=n;++i)
for(R int j=1;j<=n;++j)
if(p<=max(i,j))
dis[i][j]=min(dis[i][j],dis[i][p]+dis[p][j]);
// for(R int i=1;i<=n;++i)
// for(R int j=1;j<=n;++j)
// printf("%d%c",dis[i][j],(j==n ? '\n':' '));
for(R int i=1;i<=n;++i)
for(R int j=i+1;j<=n;++j)
add(i,j+n,1,dis[i][j]);
add(S,1,k,0);for(R int i=2;i<=n;++i) add(S,i,1,0);
for(R int i=1;i<=n;++i) add(i+n,T,1,0);
EK();
printf("%lld\n",ans);
// fclose(stdin);
// fclose(stdout);
return 0;
}
HEOI 2019 RP++
P4542 [ZJOI2011]营救皮卡丘的更多相关文章
- P4542 [ZJOI2011]营救皮卡丘(Floyd+网络流)
P4542 [ZJOI2011]营救皮卡丘 乍一看似乎没啥题相似的 仔细一看,$N<=150$ 边又是双向边,似乎可以用Floyd搞 先跑一遍Floyd处理出$dis[i][j]$ 注意到走 ...
- 洛咕P4542 [ZJOI2011]营救皮卡丘
套路题? 感觉讲不清,先写建图 把每个点拆成两个,A和B, S->Ai流量=1费用=0,Bi->T流量=1费用=0, Ai->Bj流量=1费用=ij最短路 还有一个特殊的s点,S-& ...
- bzoj 2324 [ZJOI2011]营救皮卡丘(floyd,费用流)
2324: [ZJOI2011]营救皮卡丘 Time Limit: 10 Sec Memory Limit: 256 MBSubmit: 1777 Solved: 712[Submit][Stat ...
- BZOJ2324: [ZJOI2011]营救皮卡丘
2324: [ZJOI2011]营救皮卡丘 Time Limit: 10 Sec Memory Limit: 256 MBSubmit: 1359 Solved: 522[Submit][Stat ...
- BZOJ 2324: [ZJOI2011]营救皮卡丘( floyd + 费用流 )
昨晚写的题...补发一下题解... 把1~N每个点拆成xi, yi 2个. 预处理i->j经过编号不超过max(i,j)的最短路(floyd) S->0(K, 0), S->xi(1 ...
- 【BZOJ2324】[ZJOI2011]营救皮卡丘(网络流,费用流)
[BZOJ2324][ZJOI2011]营救皮卡丘(网络流,费用流) 题面 BZOJ 洛谷 题解 如果考虑每个人走的路径,就会很麻烦. 转过来考虑每个人破坏的点集,这样子每个人可以得到一个上升的序列. ...
- 【BZOJ2324】[ZJOI2011]营救皮卡丘 有上下界费用流
[BZOJ2324][ZJOI2011]营救皮卡丘 Description 皮卡丘被火箭队用邪恶的计谋抢走了!这三个坏家伙还给小智留下了赤果果的挑衅!为了皮卡丘,也为了正义,小智和他的朋友们义不容辞的 ...
- bzoj2324 [ZJOI2011]营救皮卡丘 费用流
[ZJOI2011]营救皮卡丘 Time Limit: 10 Sec Memory Limit: 256 MBSubmit: 2653 Solved: 1101[Submit][Status][D ...
- 【BZOJ 2324】 [ZJOI2011]营救皮卡丘
Description 皮卡丘被火箭队用邪恶的计谋抢走了!这三个坏家伙还给小智留下了赤果果的挑衅!为了皮卡丘,也为了正义,小智和他的朋友们义不容辞的踏上了营救皮卡丘的道路. 火箭队一共有N个据点,据点 ...
随机推荐
- MongoDB中使用的SCRAM-SHA1认证机制
摘要: 介绍 SCRAM是密码学中的一种认证机制,全称Salted Challenge Response Authentication Mechanism. SCRAM适用于使用基于『用户名:密码』这 ...
- sql添加/移除约束
唯一:ALTER TABLE TableName ADD CONSTRAINT UQ_NickName UNIQUE(NickName) 主键:ALTER TABLE TableName ADD CO ...
- Laravel 使用 seeder 使用要点
一.关于 DB use DB; 再使用 DB::table(database.table)->get(); 二.关于 ERROR 1366 (HY000): Incorrect string v ...
- Java Thread系列(二)线程状态
Java Thread系列(二)线程状态 一.线程的五种状态 新建状态(New):新创建了一个线程对象,尚未启动. 就绪状态(Runnable):也叫可运行状态.线程对象创建后,其他线程调用了该对象的 ...
- java可视化
1.java关闭窗口代码. ft.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE); IE打开 Weiler-Atherton任意多边形裁剪 http:/ ...
- Linux 基础教程 38-文件下载
什么是wget wget用原始帮助里面的英文来讲就是:The non-interactive network downloader,非交互式网络下载器.它支持HTTP.HTTPS.FTP等协议 ...
- POJ2195 Going Home (最小费最大流||二分图最大权匹配) 2017-02-12 12:14 131人阅读 评论(0) 收藏
Going Home Description On a grid map there are n little men and n houses. In each unit time, every l ...
- [编译,报错以及其他] 有关C/C++中int不能用-2147483648当最小值的问题
这个取决于今早看耗子叔的微博: 这里说到了int的取值范围的问题,int的取值是-2147483648 ~ 2147483647,但是如果直接在编译器(VS2013)中使用-2147483648会报错 ...
- centos 下wps 与goland 不能输入中文的解决办法
输入法:CentOS7自带ibus,如果你用的是fcitx请在对应的地方进行修改 系统:CentOS7,这个方案应该适用于大多数Linux发行版本 intelliJ goland中文输入法问题解决 首 ...
- ASP.NET IIS 支持PUT、DELETE请求
IIS 本身不支持PUT.DELETE请求,但可以通过一下方法修改进而达到目的 删除IIS安装的WebDav模块,选择你的项目,右边有个“模块”,双击它:找到WebDavModule,删除它(不推荐, ...