2-SAT到图论

\(k-SAT\) 是 k-适应性问题(Satisfiability)的简称。

\(k-SAT\) 问题(除 \(k = 2\))已被证明为是 \(NP\) 完全问题, 而对于 \(k = 2\) 的 \(2-SAT\) 问题, 我们可以用图论求解。

\(k-SAT\) 与 \(2-SAT\) 最大的不同在与这个\(2\) 废话, 就因为这个 “2”, 我们可以由假设的第一个限制条件推知第二个限制条件

举个栗子: 限制条件为 \(a\) ^ $b = 1 $ (对于 \(a\) )有

\[ a\ xor\ b = 1 \Rightarrow \left\{
\begin{aligned}
a = 1 \Rightarrow b = 0 \\
a = 0 \Rightarrow b = 1
\end{aligned}
\right.
\]

依据限制条件, 我们可以搞一个通式有助后边理解: \(a = x\) \(b = y\) (a、 b、 x 已知且 \(x, y \in {0, 1}\) )

于是, 我们把问题转换到图中:引入一条有向边, 起点为通式则的左边, 终点为则右边, 这条边的意义为 必须选中, 即选择了起点就必须选择终点。

以上文的限制条件为例,我们将 x 变量为假设为 x (\(x << 1\)), x 变量为真设为 x'(\(x << 1 | 1\)) , 可以作如下转换(对于 \(a\) ):

\[ a\ xor\ b = 1 \Rightarrow \left\{
\begin{aligned}
a'\ \rightarrow\ b \\
a\ \rightarrow\ b'
\end{aligned}
\right.
\]

其代表的含义为: 选择了 a 为真则一定选择 b 为假, 选了 a 为假则一定选择 b 为真

这只是对 \(a\) 进行讨论, 事实上对于异或这一条件我们需要连 \(4\) 条边(上面两条和对于 \(b\) 的两条)

可以联想到: 在此图中沿着一条路走下去其实就是逻辑推断的过程

合法性的判断

这样我们就得到了一个图。 如何快速的判断是否能满足所有限制条件呢? 我们需要在图上做文章了。

不合法显然意味着矛盾, 什么时候会产生矛盾呢?

一个人不可能既是男的又是女的。 花是妹子, 花是男的, 这显然就矛盾了吗

我们假设 \(x\) 为假, 经过一系列推断, 我们推出了 \(x\) 为真, 这显然矛盾, 问题无解

上文提到过, 推断过程其实就是走有向边遍历的过程, 那么若是存在一个环, 同时包含了 \(x\) 和 \(x'\), 那么问题无解。

更确切的说: 当一个变量的两对立面(即 \(x\) 和 \(x'\))同时存在于一个强联通分量中时, 问题无解。

所以我们跑一次 \(Tarjan\) 判断每一元素的两对立面是否在同一强联通分量即可判断是否合法

方案输出

先导知识: 你真的了解 Tarjan 吗

你可能知道: \(Tarjan\) 可以求强联通分量, 并对在同一个强联通分量中的点染色

可你真的知道染色的顺序有什么流弊的东西吗

反向拓扑序? 拜托, 别搞那些花里胡哨的

因为 \(Tarjan\) 是利用 \(dfs\) 实现的, 说到底用的是, 先进后出, 所以对于两个点 \(u,v\) 若是两个点没有被 \(Tarjan\) 过, 且能从 \(u\) 点 到达 \(v\) 点, 那么 \(col[v] <= col[u]\) (其中 \(=\) 是两点处于一个强联通分量的时候), 即 \(v\) 比 \(u\) 先完成染色(其实就是个逆拓扑序啦)

而对于两个点, 他们无法互相通达, 那么先 \(Tarjan\) 哪一个, 那一个点的颜色编号较小

优先级

在 \(2-SAT\) 问题中, 我们用 \(Tarjan\) 不仅是求出了强联通分量, 他还进行了一个遍历整个图的过程, 也就是说, 我们利用 \(Tarjan\) 完成了逻辑推理

因为遍历有先后, 所以所对应的逻辑推理也有一个优先级(这点很重要): 试想现在有一个点 \(X\), 我们从 \(X\) 出发进行 \(Tarjan\) ,他的所有子节点(\(x_{1},x_{2}...,x_{n}\))的颜色编号小于等于此节点的颜色编号(即 \(col[X] >= col[x_{1},x_{2}...,x_{n}]\)), 我们可以认为这些子节点是由这一个点逻辑推理出来的; 好的现在我们遍历下一个点 \(Y\), 他的所有子节点(\(y_{1},y_{2}...,y_{m}\))颜色编号小于 \(col[Y]\), 所以我们可以得到一个式子:

\[col[x_{1},x_{2}...,x_{n}] <= col[X] <= col[y_{1},y_{2}...,y_{m}] <= col[Y]
\]

从中间的 \(<=\) 号处切开, 左右分类, 可以很清晰的看出他们的优先级关系, 所以我们得到结论: 颜色标号越小, 优先级越高

所以输出方案时(当然已经判断合法后)选取真和假这两个对立面中颜色编号较小的点, 他的优先级高, 我们选取他即可

几种模型

逻辑运算无非 \(or, and, xor\) 这三种, 所以模型并不是很多(依然 \(x\) 代表假, \(x'\) 代表真)

\(a\ or\ b = 0\)

即 \(a = b = 0\), 优先级 \(0 >1\)

\(a' \rightarrow a\)

\(b' \rightarrow b\)

\(a\ or\ b = 1\)

即 两个不能同时为 \(0 \Rightarrow\) 一个为 \(0\) 则另一个为 \(1\)

\(a \rightarrow b'\)

\(b \rightarrow a'\)

\(a\ and\ b = 0\)

即 两个不能同时为 \(1 \Rightarrow\) 一个为 \(1\) 则另一个为 \(0\)

\(a' \rightarrow b\)

\(b' \rightarrow a\)

\(a\ and\ b = 1\)

即 \(a = b = 1\), 优先级 \(1 > 0\)

\(a \rightarrow a'\)

\(b \rightarrow b'\)

\(a\ xor\ b = 0\)

即 两个变量相同

\(a \rightarrow b\)

\(a' \rightarrow b'\)

\(b \rightarrow a\)

\(b' \rightarrow a'\)

\(a\ xor\ b = 1\)

即 两个变量不同

\(a \rightarrow b'\)

\(a' \rightarrow b\)

\(b \rightarrow a'\)

\(b' \rightarrow a\)

P4782 【模板】2-SAT 问题

题目背景

2-SAT 问题 模板

题目描述

有n个布尔变量 x_1x ~ x_n,另有m个需要满足的条件,每个条件的形式都是“ x_i为true/false或 x_j为true/false”。比如“ x_1为真或 x_3为假”、“ x_7为假或 x_2为假”。2-SAT 问题的目标是给每个变量赋值使得所有条件得到满足。

输入输出格式

输入格式:

第一行两个整数n和m,意义如题面所述。

接下来m行每行4个整数 i a j b,表示“ x_i为a或 x_j为b”(a,b∈{0,1})

输出格式:

如无解,输出“IMPOSSIBLE”(不带引号); 否则输出"POSSIBLE"(不带引号),下 一行n个整数 x_1 ~ x_n( x_i∈{0,1}),表示构造出的解。


蛮裸的, 主要靠输出方案, 搞清楚 优先级即可

Code

#include<iostream>
#include<cstdio>
#include<queue>
#include<cstring>
#include<algorithm>
typedef long long LL;
using namespace std;
int RD(){
int out = 0,flag = 1;char c = getchar();
while(c < '0' || c >'9'){if(c == '-')flag = -1;c = getchar();}
while(c >= '0' && c <= '9'){out = out * 10 + c - '0';c = getchar();}
return flag * out;
}
const int maxn = 2000019,INF = 1e9 + 19;
int head[maxn],nume = 1;
struct Node{
int v,dis,nxt;
}E[maxn << 2];
void add(int u,int v,int dis){
E[++nume].nxt = head[u];
E[nume].v = v;
E[nume].dis = dis;
head[u] = nume;
}
int num, nr;
int DFN[maxn], LOW[maxn], INDEX;
int S[maxn], top;
bool ins[maxn];
int col[maxn], numc;
void Tarjan(int u){
DFN[u] = LOW[u] = ++INDEX;
S[++top] = u;ins[u] = 1;
for(int i = head[u];i;i = E[i].nxt){
int v = E[i].v;
if(!DFN[v])Tarjan(v), LOW[u] = min(LOW[u], LOW[v]);
else if(ins[v])LOW[u] = min(LOW[u], DFN[v]);
}
if(DFN[u] == LOW[u]){
numc++;
while(S[top + 1] != u){
col[S[top]] = numc;
ins[S[top--]] = 0;
}
}
}
int main(){
num = RD();nr = RD();
for(int i = 1;i <= nr;i++){
int a = RD(), x = RD(), b = RD(), y = RD();
//<< 1 | 0 -> 0 , << 1 | 1 -> 1
add(a << 1 | (x ^ 1), b << 1 | y, 1);
add(b << 1 | (y ^ 1), a << 1 | x, 1);
}
for(int i = 2;i <= (num << 1 | 1);i++)if(!DFN[i])Tarjan(i);
for(int i = 1;i <= num;i++){
if(col[i << 1] == col[i << 1 | 1]){
puts("IMPOSSIBLE");
return 0;
}
}
puts("POSSIBLE");
for(int i = 1;i <= num;i++){
if(col[i << 1] < col[i << 1 | 1])printf("0 ");
else printf("1 ");
}
puts("");
return 0;
}

P4782 【模板】2-SAT 问题 && 2-SAT问题的更多相关文章

  1. [洛谷P4782] [模板] 2-SAT 问题

    NOIp后第一篇题解. NOIp我考的很凉啊...... 题目传送门 之前讲过怎么判断2-SAT是否存在解. 至于如何构造一组解: 我们想到对tarjan缩点后的图进行拓扑排序. 那么对于代表0状态的 ...

  2. 浅谈2-SAT

    引入: 相信大家都了解过差分约束系统.差分约束系统的大体意思就是给出一些有某种关系的变量,问你是否有某种赋值使得这些关系全部成立 其实\(2-SAT\)的题目描述和这个很像(虽然解法不一样) 那么\( ...

  3. 《Linux命令行与shell脚本编程大全》第二十章 正则表达式

    20.1 什么是正则表达式 20.1.1 定义 正则表达式是你所定义的模式模板.linux工具可以用它来过滤文本. 正则表达式利用通配符来描述数据流中第一个或多个字符. 正则表达式模式含有文本或特殊字 ...

  4. DirectX11 With Windows SDK--11 混合状态与光栅化状态

    前言 虽然这一部分的内容主要偏向于混合(Blending),但这里还需提及一下,关于渲染管线可以绑定的状态主要有如下四种: 光栅化状态(光栅化阶段) 采样器状态(像素着色阶段) 混合状态(输出合并阶段 ...

  5. MySQL-MHA集群部署(binlog复制)

    MHA的理论知识网上有很多教程,这里不会说明:仅推荐博客链接! MHA的理论说明:http://www.ywnds.com/?p=8094 MHA的安装包需要在google上面下载,或者就是csdn上 ...

  6. DirectX11 With Windows SDK--11 混合状态

    原文:DirectX11 With Windows SDK--11 混合状态 前言 这一章会着重讲述混合状态,在下一章则会讲述深度/模板状态 DirectX11 With Windows SDK完整目 ...

  7. Enter password for default keyring to unlock

    file /home/ok/.gnome2/keyrings/login.keyring /home/ok/.gnome2/keyrings/login.keyring: GNOME keyring, ...

  8. 原来还有这样的记词方法_Java版记不规则动词_博主推荐

    昨天在看一本英语书的不规则动词的时候,突然产生的灵感:就是想把这样记单词简单方式,用程序代码实现,然后,使用户可以与之进行交互 这样,在用户背不规则动词的时候就会轻松把它给记住.基于这一点,于是我就思 ...

  9. FTP常用故障代码注解

    FTP错误列表 出处:http://bbs.enet.com.cn/UserControl?act=13&threadID 作者: |秒杀』| 详细的FTP错误列表 Restart marke ...

  10. python中数据的变量和字符串的常用使用方法

    1.查看变量类型: a=2 print(a,type(a)) print的用法:在print后面跟多个输出,可以用逗号分隔. 回收变量名,如把a存储不同的数据,你不需要删除原有变量就可以直接赋值 2. ...

随机推荐

  1. 实验一linux 系统简介和实验二基本概念及操作

    作业 zy e

  2. 20162328蔡文琛 大二week07

    20162328 2017-2018-1 <程序设计与数据结构>第7周学习总结 教材学习内容总结 树是非线性结构,其元素组织为一个层次结构. 树的度表示树种任意节点的最大子节点数. 有m个 ...

  3. GITHUB随笔 15-5月 junit

    junit 是用来做单元测试的一个工具  测试是一个持续的过程.也就是说测试贯穿与开发的整个过程中,单元测试尤其适合于迭代增量式的开发过程. @ignore:   该元数据标记的测试方法在测试中会被忽 ...

  4. Java中实现PCA降维

    package com.excellence.splitsentence; import java.net.UnknownHostException; import java.util.ArrayLi ...

  5. grunt入门讲解2:如何使用 Gruntfile 配置任务

    Grunt的task配置都是在 Gruntfile 中的grunt.initConfig方法中指定的.此配置主要包括以任务名称命名的属性,和其他任意数据.一旦这些代表任意数据的属性与任务所需要的属性相 ...

  6. node websocket学习研究

    websocket作为不同于http的数据传输方式,是开发一些实时系统的不二选择. 最近在研究开发websocket方面的小程序.小程序客户端直接对websocket做了封装.自己只要写后端就可以了. ...

  7. delphi 删除字符串的回车、空格、Tab键

    myStr:=StringReplace(myStr, chr(13)+chr(10), '', [rfReplaceAll]);//删除回车      myStr:=StringReplace(my ...

  8. 解决Linux关闭SSH,终端后运行程序终止问题(包括后台)

    问题描述: 每次SSH到服务器上,然后运行了一个自己写的服务端程序,比如 ./myserver.sh ,然后关闭ssh或者终端之后,发现服务不能访问. 简要分析下: 根据   这篇博文  的提示,ss ...

  9. 深度学习:卷积神经网络(convolution neural network)

    (一)卷积神经网络 卷积神经网络最早是由Lecun在1998年提出的. 卷积神经网络通畅使用的三个基本概念为: 1.局部视觉域: 2.权值共享: 3.池化操作. 在卷积神经网络中,局部接受域表明输入图 ...

  10. DataTable学习笔记 - 01

    DataTable 是 jQuery 的一个插件. 代码上来吧, <!DOCTYPE html> <html> <head> <meta charset=&q ...