[模板]2-SAT 问题&和平委员会
tarjan的运用
this is a problem:link
2-SAT处理的是什么
首先,把「2」和「SAT」拆开。SAT 是 Satisfiability 的缩写,意为可满足性。即一串布尔变量,每个变量只能为真或假。要求对这些变量进行赋值,满足布尔方程。
所以看这道题
若ai为真或aj为真,所以当ai为真时aj必须为假,若aj为真时ai必须为假
所以假设i为ai为真,i+n为ai为假
所以建边(i,j+n),(j,i+n),连接u,v表示选u就要选v
然后用可能出现环,所以用tarjan缩点表示若i,j在同一强联通分量中,那么他们两个必须要选
然后可以想无解情况,若i于i+n都在同一强连通分量中,于是无解,因为他们两个都必须要选,但却不能都选
然后想怎么输出解
因为拓扑序的关系 拓扑序在前的点有可能能够到达拓扑序靠后的点 而靠后的一定无法到达靠前的 因此选择靠后的一定能够保证出合法解 而选择靠前的可能会导致前面的点存在一条通路到达后面的点致使答案出现错误 这样可以感性理解,至于证明利用了建图过程中的对称性。什么意思呢1,就是tarjan为反拓扑顺序,所以说就要选择强连通编号中最大的
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
inline int read()
{
int f=,ans=;char c;
while(c<''||c>''){if(c=='-')f=-;c=getchar();}
while(c>=''&&c<=''){ans=ans*+c-'';c=getchar();}
return f*ans;
}
struct node{
int u,v,nex;
}x[];
int n,m,head[],cnt,dfn[],st[],num,top,col,co[],low[],tot;
void tarjan(int xx)
{
low[xx]=dfn[xx]=++num;
st[++tot]=xx;
for(int i=head[xx];i!=-;i=x[i].nex)
{
int v=x[i].v;
if(!dfn[v])
{
tarjan(v);
low[xx]=min(low[xx],low[v]);
}
else if(!co[v]) low[xx]=min(low[xx],dfn[v]);
}
if(dfn[xx]==low[xx])
{
co[xx]=++col;
while(st[tot]!=xx)
{
co[st[tot]]=col;
tot--;
}
tot--;
}
return;
}
void add(int u,int v)
{
x[cnt].u=u,x[cnt].v=v,x[cnt].nex=head[u],head[u]=cnt++;
}
int main()
{
memset(head,-,sizeof(head));
n=read(),m=read();
for(int i=;i<=m;i++)
{
int a=read(),xa=read(),b=read(),yb=read();
if(xa==&&yb==) add(a,b+n),add(b,a+n);
if(xa==&&yb==) add(a,b),add(b+n,a+n);
if(xa==&&yb==) add(a+n,b),add(b+n,a);
if(xa==&&yb==) add(a+n,b+n),add(b,a);
}
for(int i=;i<=*n;i++)
if(!dfn[i]) tarjan(i);
for(int i=;i<=n;i++)
if(co[i]==co[i+n]){cout<<"IMPOSSIBLE";return ;}
cout<<"POSSIBLE"<<endl;
for(int i=;i<=n;i++)
{
if(co[i]>co[i+n]) cout<<<<" ";
else cout<<<<" ";
}
return ;
}
/*
3 2
1 1 3 0
3 1 1 0
*/
一道2-SAT的例题
和平委员会 |
| link |
|
试题描述
|
|
根据宪法,Byteland 民主共和国的公众和平委员会应该在国会中通过立法程序来创立。 不幸的是,由于某些党派代表之间的不和睦而使得这件事存在障碍。
此委员会必须满足下列条件: 每个党派都在委员会中恰有 1 个代表, 如果 2 个代表彼此厌恶,则他们不能都属于委员会。 每个党在议会中有 2 个代表。代表从 1 编号到 2n。 编号为 2i−1 和 2i 的代表属于第 i 个党派。 任务:写一程序读入党派的数量和关系不友好的代表对,计算决定建立和平委员会是否可能,若行,则列出委员会的成员表。 |
|
输入
|
|
第一行有两个非负整数 n 和 m。他们各自表示:党派的数量 n 和不友好的代表对 m。 接下来 m 行,每行为一对整数 a,b,表示代表 a,b 互相厌恶。
|
|
输出
|
|
如果不能创立委员会,则输出信息NIE。若能够成立,则输出包括 n 个从区间 1 到 2n 选出的整数,按升序写出,每行一个,这些数字为委员会中代表的编号。
如果委员会能以多种方法形成,程序可以只输出它们的某一个。 |
|
输入示例
|
|
3 2
1 3 2 4 |
|
输出示例
|
|
1
4 5 |
|
其他说明
|
|
数据范围与提示
1≤n≤8000,0≤m≤20000,1≤a<b≤2n |
一个2-SAT的板子,就是与上一道题的区别是,上一道题必须二选一,这一次可以两个都不选,所以选择拓扑序小的(想一想,为什么)、
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
inline int read()
{
int f=,ans=;char c;
while(c<''||c>''){if(c=='-')f=-;c=getchar();}
while(c>=''&&c<=''){ans=ans*+c-'';c=getchar();}
return f*ans;
}
struct node{
int u,v,nex;
}x[];
int n,m,cnt,head[],dfn[],low[],tot,num,st[],co[],col;
void add(int u,int v)
{
x[cnt].u=u,x[cnt].v=v,x[cnt].nex=head[u],head[u]=cnt++;
}
int cx(int s)
{
if(s%==) return s-;
else return s+;
}
void tarjan(int xx)
{
low[xx]=dfn[xx]=++num;
st[++tot]=xx;
for(int i=head[xx];i!=-;i=x[i].nex)
{
int v=x[i].v;
if(!dfn[v])
{
tarjan(v);
low[xx]=min(low[xx],low[v]);
}
else if(!co[v]) low[xx]=min(low[xx],dfn[v]);
}
if(dfn[xx]==low[xx])
{
co[xx]=++col;
while(st[tot]!=xx)
{
co[st[tot]]=col;
tot--;
}
tot--;
}
return;
}
int main()
{
memset(head,-,sizeof(head));
n=read(),m=read();
for(int i=;i<=m;i++)
{
int u=read(),v=read();
add(u,cx(v)),add(v,cx(u));
}
for(int i=;i<=*n;i++)
if(!dfn[i]) tarjan(i);
for(int i=;i<=*n;i=i+)
{
if(co[i]==co[i+])
{
cout<<"NIE";
return ;
}
}
for(int i=;i<=*n;i=i+)
{
if(co[i]>co[i+]) cout<<i+<<endl;
else cout<<i<<endl;
}
return ;
}
[模板]2-SAT 问题&和平委员会的更多相关文章
- HDU 1814 Peaceful Commission / HIT 1917 Peaceful Commission /CJOJ 1288 和平委员会(2-sat模板题)
HDU 1814 Peaceful Commission / HIT 1917 Peaceful Commission /CJOJ 1288 和平委员会(2-sat模板题) Description T ...
- 【POI2001】【HDU1814】和平委员会
题面 Description 根据宪法,Byteland民主共和国的公众和平委员会应该在国会中通过立法程序来创立. 不幸的是,由于某些党派代表之间的不和睦而使得这件事存在障碍. 此委员会必须满足下列条 ...
- COGS:313. [POI2001] 和平委员会
313. [POI2001] 和平委员会 ★★☆ 输入文件:spo.in 输出文件:spo.out 评测插件时间限制:1 s 内存限制:128 MB 题目描述 根据宪法,Bytelan ...
- HDU1814和平委员会
题目大意: 有n对的人,编号从1-2*n,m对的人之间互相不喜欢,每对人中必徐选1个人加入和平委员会,求字典序最小的解 -------------------------------- 2-SAT问题 ...
- [POI2001]和平委员会
题目描述 根据宪法,Byteland民主共和国的公众和平委员会应该在国会中通过立法程序来创立. 不幸的是,由于某些党派代表之间的不和睦而使得这件事存在障碍. 此委员会必须满足下列条件: 每个党派都在委 ...
- LOJ10097和平委员会
POI 2001 根据宪法,Byteland民主共和国的公众和平委员会应该在国会中通过立法程序来创立. 不幸的是,由于某些党派代表之间的不和睦而使得这件事存在障碍. 此委员会必须满足下列条件: 每个党 ...
- cogs 313. [POI2001] 和平委员会(2-SAT
http://cogs.pro:8080/cogs/problem/problem.php?pid=pyzQimjkj 题意:有n个集合,每个集合有俩元素,要从n个中各选一个放一堆,但是有的俩不能同时 ...
- P5782-[POI2001]和平委员会【2-SAT】
正题 题目链接:https://www.luogu.com.cn/problem/P5782 题目大意 \(n\)对人,每对之间恰好有一个人出席.\(m\)对仇恨关系表示两个人不能同时出席. 求是否有 ...
- HIT 1917 2—SAT
题目大意:一国有n个党派,每个党派在议会中都有2个代表, 现要组建和平委员会,要从每个党派在议会的代表中选出1人,一共n人组成和平委员会. 已知有一些代表之间存在仇恨,也就是说他们不能同时被选为和平委 ...
随机推荐
- Visual Studio设置字体及护眼背景色
打开vs 菜单栏选择: 工具 -> 选择 -> 环境 -> 字体和颜色,如图所示 字体可以如上选择,背景色选择项背景,点击自定义,如下设置即可.
- JSP整理
JSP全称Java Server Pages,是一种动态网页开发技术.它使用JSP标签在HTML网页中插入Java代码.标签通常以<%开头以%>结束. JSP是一种Java servlet ...
- 小白初识 - 快速排序(QuickSort)
我个人觉得快速排序和归并排序有相似之处,都是用到了分治的思想,将大问题拆分成若干个小问题. 不同的地方是归并排序是先把大问题拆分好了之后再排序,而快速排序则是一边拆分,一边排序. 快速排序的原理就是, ...
- 实现短信超链接调起APP
因APP推广的需求,需要给APP用户定期发送短信提醒登录使用,为了更好的用户体验在短信内容中嵌入了可以直接打开APP的超链接,下面介绍一下具体的代码实现. 编辑openApp.html文件: < ...
- [转]oracle数据库定时任务dbms_job的用法详解
这篇文章给大家详细介绍了dbms_job的用法,用于安排和管理作业队列,通过使用作业,可以使ORACLE数据库定期执行特定的任务.有需要的朋友们可以参考借鉴. 一.dbms_job涉及到的知识点 ...
- Python学习之路6 - 装饰器
装饰器 定义:本质是函数,(装饰其他函数)就是为其他函数添加附加功能.原则:1.不能修改被装饰的函数的源代码 2.不能修改被装饰的函数的调用方式 实现装饰器的知识储备: 1.函数即“变量” 2.高阶函 ...
- 静默调用ShellContextMenu 实现QQ文件共享
我在CSDN提问题一直没人回复,一下午时间自己终于解决了问题 http://bbs.csdn.net/topics/391916381 现将过程录下 先说需求,我想实现的功能是 在程序中对文件调用百度 ...
- WPF和Expression Blend开发实例:Adorner(装饰器)应用实例
装饰器-- 表示用于修饰 UIElement 的 FrameworkElement 的抽象类 简单来说就是,在不改变一个UIElement结构的情况下,将一个Visual对象加到它上面. 应用举例: ...
- Redis 简要介绍--用于讲解消息中间件
1:安装 Redis yum install -y redis 2:编辑配置文件/etc/redis.conf,Redis作为一个消息中间件,那么应该监听于本机的外网socket上,因此修改 bi ...
- HDU 2164 Rock, Paper, or Scissors?
http://acm.hdu.edu.cn/showproblem.php?pid=2164 Problem Description Rock, Paper, Scissors is a two pl ...