Tarjan/2-SAT学习笔记
Tarjan/2-SAT
Tags:图论
作业部落
评论地址
Tarjan
用来求割边或者割点,求点双联通分量或者边双联通分量
点双联通分量:两个点之间有两条点不相交的路径
边双联通分量:两个点之间有两条边不相交的路径
Tarjan求LCA还不会
2-SAT
每种物品有选或者不选两种状态,有些限制条件形如
选了\(A\)则必须选\(B\),\(A\)和\(B\)不能同时选,必须选\(A\)等等
把逻辑限制关系变成连边
a->b
表示如果\(a\)成立那么\(b\)一定成立
这个要求你理解逆否命题
逆否命题,举个例子,选\(A\)必须选\(B\),那么我选了\(B'\)就不能选\(A\),选\(B'\)就必须选\(A'\)
由于逆否命题产生的对称性使得\(2-SAT\)问题得以在\(O(n)\)时间求解
具体来说要求同时连接x->y
y'->x'
这样跑一遍\(Tarjan\)缩点后如果统一物品的两种状态在同一个边双连通分量中就无解
否则可以输出方案,具体来说是每个点选择\(Tarjan\)缩成的超级点中编号最小的那个(也就是反图拓扑序最小的那个)
题单
Tarjan
- [x] [luogu3387]【模板】缩点 https://www.luogu.org/problemnew/show/P3387
- [x] [luogu3388]【模板】割点(割顶)https://www.luogu.org/problemnew/show/P3388
- [x] [APIO2009]抢掠计划 https://www.luogu.org/problemnew/show/P3627
- [x] [HAOI2006]受欢迎的牛 https://www.luogu.org/problemnew/show/P2341
- [ ] [USACO5.3]校园网 https://www.luogu.org/problemnew/show/P2746
- [ ] [luogu3398]仓鼠找sugar https://www.luogu.org/problemnew/show/P3398
- [x] [HNOI2006]潘多拉的宝盒 https://www.luogu.org/problemnew/show/P2321
- [ ] [HAOI2010]软件安装 https://www.luogu.org/problemnew/show/P2515
- [x] [HNOI2012]矿场搭建 https://www.luogu.org/problemnew/show/P3225
- [ ] [SDOI2010]所驼门王的宝藏 https://www.luogu.org/problemnew/show/P2403
2-SAT
- [x] [POI2001]和平委员会 http://cogs.pro:8080/cogs/problem/problem.php?pid=313
- [x] [JSOI2010]满汉全席 https://www.luogu.org/problemnew/show/P4171
- [x] [UOJ210]寻找罪犯 http://uoj.ac/problem/210
- [x] [NOI2017]游戏 https://www.luogu.org/problemnew/show/P3825
- [ ] [HNOI2010]平面图判定 https://www.luogu.org/problemnew/show/P3209
- [ ] [POI2010]KOL-Railway https://www.luogu.org/problemnew/show/P3497
Code
边双
void Tarjan(int x)
{
vis[x]=1;sta[++top]=x;
dfn[x]=low[x]=++tot;
for(int i=A.head[x],R=A.a[i].to;i;i=A.a[i].next,R=A.a[i].to)
if(!dfn[R]) Tarjan(R),low[x]=min(low[x],low[R]);
else if(vis[R]) low[x]=min(low[x],low[R]);
if(low[x]!=dfn[x]) return;
for(int k=sta[top],lst=0;lst!=x;lst=k,k=sta[--top])
vis[k]=0,bel[k]=x,val[x]+=val[k]*(k!=x);
}
点双
void Tarjan(int x,int f)
{
int s=0;dfn[x]=low[x]=++tot;
for(int i=head[x];i;i=a[i].next)
{
int R=a[i].to;if(R==f) continue;
if(dfn[R]) {low[x]=min(low[x],dfn[R]);continue;}
s++;Tarjan(R,x);tag[x]|=low[R]>=dfn[x];
low[x]=min(low[x],low[R]);
}
if(!f&&s==1) tag[x]=0;
}
!注意点双第7行一定要是dfn[R]
2-sat(和平委员会)
#include<cstdio>
#include<cstdlib>
#include<queue>
using namespace std;
const int N=41000;
struct edge {int next,fr,to;};
struct Map
{
edge a[N]; int head[N],cnt;
void link(int x,int y) {a[++cnt]=(edge){head[x],x,y};head[x]=cnt;}
}A,B;
int dfn[N],top,sta[N],vis[N],low[N];
int bel[N],tot,n,m,p[N],node;
queue<int> Q;
void Min(int &a,int b) {if(b<a) a=b;}
void Tarjan(int x)
{
vis[x]=1;sta[++top]=x; dfn[x]=low[x]=++tot;
for(int i=A.head[x],R=A.a[i].to;i;i=A.a[i].next,R=A.a[i].to)
if(!dfn[R]) Tarjan(R),Min(low[x],low[R]);
else if(vis[R]) Min(low[x],low[R]);
if(low[x]!=dfn[x]) return;++node;
for(int k=sta[top],lst=0;lst!=x;lst=k,k=sta[--top])
vis[k]=0,bel[k]=node;
}
int main()
{
// freopen("spo.in","r",stdin);
// freopen("spo.out","w",stdout);
scanf("%d%d",&n,&m);
for(int i=1;i<=n*2;i++) p[i]=i&1?i+1:i-1;
for(int i=1;i<=m;i++)
{
int x,y;scanf("%d%d",&x,&y);
A.link(x,p[y]);A.link(y,p[x]);
}
for(int i=1;i<=n*2;i++) if(!dfn[i]) Tarjan(i);
for(int i=1;i<=n*2;i+=2) if(bel[i]==bel[p[i]]) {puts("NIE");exit(0);}
for(int i=1;i<=n*2;i+=2) printf("%d\n",bel[i]<bel[p[i]]?i:p[i]);
}
Tarjan/2-SAT学习笔记的更多相关文章
- $tarjan$简要学习笔记
$QwQ$因为$gql$的$tarjan$一直很差所以一直想着要写个学习笔记,,,咕了$inf$天之后终于还是写了嘻嘻. 首先说下几个重要数组的基本定义. $dfn$太简单了不说$QwQ$ 但是因为有 ...
- <老友记>学习笔记
这是六个人的故事,从不服输而又有强烈控制欲的monica,未经世事的千金大小姐rachel,正直又专情的ross,幽默风趣的chandle,古怪迷人的phoebe,花心天真的joey——六个好友之间的 ...
- OGG学习笔记02-单向复制配置实例
OGG学习笔记02-单向复制配置实例 实验环境: 源端:192.168.1.30,Oracle 10.2.0.5 单实例 目标端:192.168.1.31,Oracle 10.2.0.5 单实例 1. ...
- python数据分析入门学习笔记
学习利用python进行数据分析的笔记&下星期二内部交流会要讲的内容,一并分享给大家.博主粗心大意,有什么不对的地方欢迎指正~还有许多尚待完善的地方,待我一边学习一边完善~ 前言:各种和数据分 ...
- 仙人掌&圆方树学习笔记
仙人掌&圆方树学习笔记 1.仙人掌 圆方树用来干啥? --处理仙人掌的问题. 仙人掌是啥? (图片来自于\(BZOJ1023\)) --也就是任意一条边只会出现在一个环里面. 当然,如果你的图 ...
- 【MarkMark学习笔记学习笔记】javascript/js 学习笔记
1.0, 概述.JavaScript是ECMAScript的实现之一 2.0,在HTML中使用JavaScript. 2.1 3.0,基本概念 3.1,ECMAScript中的一切(变量,函数名,操作 ...
- Linux 学习笔记之超详细基础linux命令 Part 13
Linux学习笔记之超详细基础linux命令 by:授客 QQ:1033553122 ---------------------------------接Part 12---------------- ...
- Linux 学习笔记之超详细基础linux命令 Part 8
Linux学习笔记之超详细基础linux命令 by:授客 QQ:1033553122 ---------------------------------接Part 7----------------- ...
- Deep learning with Python 学习笔记(5)
本节讲深度学习用于文本和序列 用于处理序列的两种基本的深度学习算法分别是循环神经网络(recurrent neural network)和一维卷积神经网络(1D convnet) 与其他所有神经网络一 ...
- 【Redis】命令学习笔记——字符串(String)(23个超全字典版)
Redis支持五种数据类型:string(字符串),hash(哈希),list(列表),set(集合)及zset(sorted set:有序集合). 本篇基于redis 4.0.11版本,学习字符串( ...
随机推荐
- Axure中移动端原型设计方法(附IPhoneX和IPhone8最新模板)
Axure中移动端原型设计方法(附IPhoneX和IPhone8最新模板) 2018年4月16日luodonggan Axure中基于设备模板的移动端原型设计方法(附IPhoneX和IPhone8最新 ...
- [WinCE | VS2008 | Solution] VS2008 building WinCE projects taking a long time
1. Open C:\Windows\Microsoft.NET\Framework\v3.5\Microsoft.CompactFramework.Common.targets 2. Find pa ...
- 关于Vue中:key="index"的console警告
在写vue项目时,浏览器的console出现如下警告信息: [Vue warn]: Property or method "index" is not defined on the ...
- 细说C#继承
简介 继承(封装.多态)是面向对象编程三大特性之一,继承的思想就是摈弃代码的冗余,实现更好的重用性. 继承从字面上理解,无外乎让人想到某人继承某人的某些东西,一个给一个拿.这个语义在生活中,就像 家族 ...
- 发现微信支付bug
第一张银行卡支付金额不足无法付款,选择另一张同样密码的银行卡,居然不用重新输入密码即可直接付款成功!
- JAVA容器全面总结
1 容器体系图 简图: 详图: 2 基础 2.1 Iterator接口 迭代器. 具有的能力:后向迭代.删除. 2.2 Iterable接口 表示一个类具有迭代 ...
- ASP.NET Web Api vs Node.js Benchmark
http://mikaelkoskinen.net/post/asp-net-web-api-vs-node-js-benchmark ASP.NET Web Api vs Node.js Bench ...
- python多进程(一)
操作系统进程 Unix/Linux操作系统提供了一个fork()系统调用,它非常特殊.普通的函数调用,调用一次,返回一次,但是fork()调用一次,返回两次,因为操作系统自动把当前进程(称为父进程)复 ...
- 在centos上安装smplayer播放器
Smplayer是一个非常好用的媒体播放器.可以支持大部分的视频和音频文件.它支持音频轨道切换.允许调节亮度.对比度.色调.饱和度.伽玛值,按照倍速.4倍速等多种速度回放.还可以进行音频和字幕延迟调整 ...
- 2.2 IPython基础
2.2 IPython基础 1 Running the IPython Shell (运行IPython Shell) 可以通过命令行启动IPython,就像启动标准的Python解释器一样,直接在t ...