Codeforces 1027D Mouse Hunt (强连通缩点 || DFS+并查集)
<题目链接>
题目大意:
有n个房间,每个房间都会有一只老鼠。处于第i个房间的老鼠可以逃窜到第ai个房间中。现在要清理掉所有的老鼠,而在第i个房间中防止老鼠夹的花费是ci,问你消灭掉所有老鼠的最少花费。
解题分析:
首先就是要注意老鼠的逃生路线为强连通分量的情况,毫无疑问,这种情况就是在那个强连通分量中的代价最小的房间安装老鼠夹(因为根据老鼠的流通性,只需要在连通分量中安装一个老鼠夹就能捕获所有的老鼠),所以我们先用Tarjan对这些房间进行缩点。然后我们只需要将那些出度为0的强连通分量的代价相加就行(强连通分量的代价=其中任意点的最小代价),因为出度为0的强连通分量一定要消除本身的老鼠,并且根据老鼠的流通性,其它出度不为0的老鼠一定也会到达这些出度为0的连通分量中。
#include <bits/stdc++.h>
using namespace std;
const int INF =0x3f3f3f3f;
#define N int(2e5+7)
#define rep(i,s,t) for(int i=s;i<=t;i++)
#define srep(i,s,t) for(int i=s;i<t;i++)
#define clr(a,b) memset(a,b,sizeof(a))
int n,top,ans,tot,scc;
int outdeg[N],to[N],stk[N],instk[N],bel[N],dfn[N],low[N],cost[N],val[N];
template<typename T> //读入优化
inline T read(T&x){
x=;;char ch=getchar();
') f|=(ch=='-'),ch=getchar();
+ch-',ch=getchar();
return x=f?-x:x;
}
void Tarjan(int u){
dfn[u]=low[u]=++tot;
stk[++top]=u;instk[u]=;
int v=to[u];
if(!dfn[v]){
Tarjan(v);
low[u]=min(low[u],low[v]);
}else if(instk[v])low[u]=min(low[u],dfn[v]);
if(dfn[u]==low[u]){
++scc;
while(true){
int v=stk[top--];
bel[v]=scc;
if(u==v)break;
}
}
}
int main(){
read(n);
rep(i,,n)read(val[i]);
rep(u,,n){
int v;read(v);to[u]=v;
}
rep(i,,n) if(!dfn[i]){ //Tarjan缩点
Tarjan(i);
}
rep(i,,scc)cost[i]=INF;
rep(i,,n)cost[bel[i]]=min(cost[bel[i]],val[i]); //cost[i]为每个强连通分量中花费最少的点
rep(u,,n){
int v=to[u];
if(bel[u]!=bel[v])outdeg[bel[u]]++; //标记每个强连通分量的出度
}
rep(i,,scc)if(!outdeg[i])ans+=cost[i]; //出度为0的点一定要安装(因为至少要消灭这些点本来的老鼠),并且只用安装出度为0的点,因为老鼠具有传递性,最终所有老鼠一定会经过这些出度为0的点
printf("%d\n",ans);
}
另一种做法,用并查集判环,DFS找环上的最小代价:
#include<bits/stdc++.h>
using namespace std;
;
int to[N], val[N], pre[N], x[N];
int find(int x){
return x==pre[x]?x:pre[x]=find(pre[x]);
}
int dfs(int x, int root){
if(x==root) return val[x]; //如果是自环,直接返回本身
return min(dfs(to[x], root), val[x]); //寻找环上权值最小的点
}
int main(){
int n;scanf("%d", &n);
;i<=n;i++)scanf("%d", &val[i]), pre[i] = i;
;i<=n;i++)scanf("%d", &to[i]);
;i<=n;i++){ //并查集判独立的连通块(判环或者判独立的点)
; //如果这个点存在环
else pre[find(to[i])] = find(i);
}
;
;i<=n;i++) if(x[i])ans+=dfs(to[i],i);
printf("%d\n",ans);
}
2019-02-18
Codeforces 1027D Mouse Hunt (强连通缩点 || DFS+并查集)的更多相关文章
- Codeforces B. Mouse Hunt(强连通分解缩点)
题目描述: Mouse Hunt time limit per test 2 seconds memory limit per test 256 megabytes input standard in ...
- CodeForces - 455C Civilization (dfs+并查集)
http://codeforces.com/problemset/problem/455/C 题意 n个结点的森林,初始有m条边,现在有两种操作,1.查询x所在联通块的最长路径并输出:2.将结点x和y ...
- 51nod 1456【强连通,缩点,并查集】
话说这道题的机遇是看到了http://blog.csdn.net/u010885899/article/details/50611895很有意思:然后就去补了这题 题意: 建最少的边使得给出的点相连. ...
- DFS/并查集 Codeforces Round #286 (Div. 2) B - Mr. Kitayuta's Colorful Graph
题目传送门 /* 题意:两点之间有不同颜色的线连通,问两点间单一颜色连通的路径有几条 DFS:暴力每个颜色,以u走到v为结束标志,累加条数 注意:无向图 */ #include <cstdio& ...
- Codeforces 745C:Hongcow Builds A Nation(并查集)
http://codeforces.com/problemset/problem/744/A 题意:在一个图里面有n个点m条边,还有k个点是受限制的,即不能从一个受限制的点走到另外一个受限制的点(有路 ...
- Codeforces Educational Codeforces Round 5 C. The Labyrinth 带权并查集
C. The Labyrinth 题目连接: http://www.codeforces.com/contest/616/problem/C Description You are given a r ...
- Codeforces Round #245 (Div. 2) B. Balls Game 并查集
B. Balls Game Time Limit: 1 Sec Memory Limit: 256 MB 题目连接 http://codeforces.com/contest/430/problem ...
- Codeforces Round #345 (Div. 1) E. Clockwork Bomb 并查集
E. Clockwork Bomb 题目连接: http://www.codeforces.com/contest/650/problem/E Description My name is James ...
- POJ3694 Network 边双缩点+LCA+并查集
辣鸡错误:把dfs和ldfs搞混...QAQ 题意:给定一个无向图,然后查询q次,求每次查询就在图上增加一条边,求剩余割边的个数. 先把边双缩点,然后预处理出LCA的倍增数组: 然后加边时,从u往上跳 ...
随机推荐
- Confluence 6 升级你的许可证
如果你修改了你的许可证(例如为你的许可证增加了更多的用户),或者从 Cloud 中整合到你本地,你需要更新你的许可证. 希望更新你的额许可证: 进入 > 基本配置(General Config ...
- Jquery无刷新实时更新表格数据
html代码: <style> .editbox { display:none } .editbox { font-size:14px; width:70px; background-co ...
- http超文本协议
当今web程序的开发技术真是百家争鸣,ASP.NET, PHP, JSP,Perl, AJAX 等等. 无论Web技术在未来如何发展,理解Web程序之间通信的基本协议相当重要, 因为它让我们理解了We ...
- Python实操
有两个列表,分别存放报名学习linux和python课程的学生名字 linux=['钢弹','小壁虎','小虎比','alex','wupeiqi','yuanhao'] python=['drago ...
- node.js 框架express关于报错页面的配置
1.声明报错的方法,以及相对应的页面 //把数据库的调用方法配置到请求中 server.use((req, res, next) => { //把数据库存入req中 req.db = db; / ...
- Python if __name__ == '__main__':(以主程序形式执行)
在外部调用某个模块时,可能会将只能在本模块执行的代码给执行了,有没有什么办法让某些特定的代码指定只能在自身运行时才执行被调用时不执行呢?使用if __name__ == '__main__':. 示例 ...
- easyui 布局之window和panel一起使用时,拉动window宽高时panel不跟随一起变化
项目开发中布局是每一个组件都由最外层的window和内部的至少一个panel组成,其他的细小组件再依次放到panel中. 问题:当拉动外部的window时我们希望内部的panel的宽高也跟着变化,但是 ...
- OpenCV-Python入门教程6-Otsu阈值法
在说Otsu之前,先说几个概念 灰度直方图:将数字图像中的所有像素,按照灰度值的大小,统计其出现的频率.其实就是每个值(0~255)的像素点个数统计. Otsu算法假设这副图片由前景色和背景色组成,通 ...
- 学习笔记: 反射应用、原理,完成扩展,emit动态代码
using Ruanmou.DB.Interface; using Ruanmou.DB.MySql; using Ruanmou.DB.SqlServer; using Ruanmou.Model; ...
- [转] Mongoose初使用总结
连接mongoose mongoose连接数据库有两种方式 第一种: 'use strict'; const mongoose = require('mongoose'); mongoose.conn ...