2-SAT超入门讲解
Preface
说实话2-SAT的题目我都没怎么做过,所以这里讲的都是些超入门什么的
还有一些板子题,由于是暑假的时候学的所以有些我也记不清了
主要学习参考自:Mancher的课件&&dalao's blog&&Another dalao
What is 2_SAT?
SAT是适定性(Satisfiability)问题的简称 。一般形式为k-适定性问题,简称 k-SAT。
可以证明,当\(k>2\)时,k-SAT是NP完全的。因此一般讨论的是\(k=2\)的情况,即2-SAT问题。
我们通俗的说,就是给你\(n\)个变量\(a_i\),每个变量能且只能取\(0/1\)的值。同时给出若干条件,形式诸如\((not)a_i\operatorname{opt}(not)\ a_j=0/1\),其中\(opt\)表示\(and,or,xor\)中的一种
而求解2-SAT的解就是求出满足所有限制的一组\(a\)
Change 2-SAT into Graph Theory
首先我们考虑将2-SAT问题往图论的方向靠,我们发现每个点要么取\(0\),要么取\(1\)。因此对于\(a_i\),我们建两个点\(2i-1\)与\(2i\)分别表示\(a_i\)取\(0\)和\(1\)
然后我们考虑建边来表示这些关系,我们令一条有向边的意义:\(x\to y\)表示如果选择了\(x\)就必须选\(y\)
那么我们可以举一些简单的例子来总结下连边的规律(用\(i'\)表示\(i\)的反面):
- \(i,j\)不能同时选:选了\(i\)就要选\(j'\),选\(j\)就要选\(i'\)。故\(i\to j',j\to i'\)。一般操作即为\(a_i \operatorname{xor} a_j=1\)
- \(i,j\)必须同时选:选了\(i\)就要选\(j\),选\(j\)就要选\(i\)。故\(i\to j,j\to i\)。一般操作即为\(a_i \operatorname{xor} a_j=0\)
- \(i,j\)只且只且选一个:选了\(i\)就要选\(j'\),选\(j\)就要选\(i'\),选\(i'\)就要选\(j\),选\(j'\)就要选\(i\)。故\(i\to j',j\to i',i'\to j,j'\to i\)。一般操作即为\(a_i \operatorname{or} a_j=1\)
- \(i\)必须选:直接\(i'\to i\),可以保证无论怎样都选\(i\)。一般操作为给出的\(a_i=1\)或\(a_i \operatorname{and} a_j=1\)
建好图然后就是考虑怎么用图论的方式解决2-SAT了。
How to solve 2-SAT——DFS
- 对于每个当前不确定的变量\(a_i\),令\(a_i=0\)然后沿着边DFS访问相连的点。
- 检查如果会导致任意一个\(j\)与\(j'\)都被选,那么撤销。否则令\(a_i=0\)
- 否则令\(a_i=1\),重复2。如果还不行就无解。
- 继续考虑下一个不确定的变量
这样的话正确性显然,由于这里的DFS涉及到全局,因此复杂度是\(O(n(n+m))\)的。
一般情况下已经很优秀了,而且还可以改进:
只需要在DFS之前判断\(i'\)能否走到\(i\)就可以省略撤销标记的过程,所以我们可以bitset优化传递闭包做到\(O(\frac{n+m}{w})\)预处理,然后就可以\(O(n+m)\)的DFS了。
这种做法还可以保证解的字典序,有时不失为一种不错的方法。
How to solve 2-SAT——SCC
考虑我们上面的判断有无解的情况,我们想到完全可以借助SCC来判断两个点是否互相到达。
那么我们先缩点,如果\(i\)与\(i'\)在同一SCC里那么显然无解。
否则选择\(i\)与\(i'\)中拓扑序较大的一个就可以得到一组可行解。
不是很常用(主要是一般题目的数据范围都不需要),用来判可行性比较好。
两道例题
- HDU 3062Party 2-SAT建图后判断可行性即可。
- HDU 1814Peaceful Commission 2-SAT建图后求字典序最小解,就用DFS的方法不预处理也可以过。
Postscript
这真的是一篇超入门博客,没有涉及特别多的难点以及姿势。
像数据结构优化建图我是肯定不会的啦,最后推荐一道比较有难度的2-SAT好题:Luogu P3825 [NOI2017]游戏&&Sol
2-SAT超入门讲解的更多相关文章
- Mysql C语言API编程入门讲解
原文:Mysql C语言API编程入门讲解 软件开发中我们经常要访问数据库,存取数据,之前已经有网友提出让鸡啄米讲讲数据库编程的知识,本文就详细讲解如何使用Mysql的C语言API进行数据库编程. ...
- 超细讲解Django打造大型企业官网
本文为知了课堂黄勇老师讲的<超细讲解Django打造大型企业官网>的笔记. 第一章 Django预热 1.创建virtualenv虚拟环境 2.URL组成部分详解 3.Django介绍 4 ...
- #001 CSS快速入门讲解
CSS入门讲解 HTML人+CSS衣服+JS动作=>DHTML CSS: 层叠样式表 CSS2.0 和 CSS3.0 版本,目前学习CSS2, CSS3只是多了一些样式出来而已 CSS 干啥用的 ...
- HTML5游戏开发引擎Pixi.js新手入门讲解
在线演示 本地下载 这篇文章中,介绍HTML5游戏引擎pixi.js的基本使用. 相关代码如下: Javascript 导入类库:(使用极客的cdn服务:http://cdn.gbtags.com) ...
- AngularJS入门讲解4:多视图,事件绑定,$resource服务讲解
上一课,大家知道,手机详细模板我们没有写出来,使用的是一个占位模板. 这一课,我们先实现手机详细信息视图,这个视图会在用户点击手机列表中的一部手机时被显示出来. 为了实现手机详细信息视图,我们将会使用 ...
- poj2104 k-th number 主席树入门讲解
poj2104 k-th number 主席树入门讲解 定义:主席树是一种可持久化的线段树 又叫函数式线段树 刚开始学是不是觉得很蒙逼啊 其实我也是 主席树说简单了 就是 保留你每一步操作完成之后 ...
- #001 HTML快速入门讲解
整理了下最早开始学习技术的笔记 3W1H学习法? (其他技术同理) What HTML是什么? When 什么时候使用HTML? Why 为什么使用HTMl? HOW 怎么使用HTML ...
- 原生AJAX入门讲解(含实例)
相对于jQuery.YUI以及其他一些类库的AJAX封装,原生JS的AJAX显得那么的尴尬,兼容性不好,要记很多的方法属性,调用不便捷,代码臃肿...但我还是想说,原生JS才是最根本最底层的知识(虽然 ...
- win8.1上wamp环境中利用apache自带ab压力测试工具使用超简单讲解
2015.10.4apache自带ab压力测试工具使用:本地环境:win8.1 wampserver2.5 -Apache-2.4.9-Mysql-5.6.17-php5.5.12-64b 可以参考一 ...
随机推荐
- Android联网更新应用
UpdateInfo public class UpdateInfo { public String version;//服务器的最新版本值 public String apkUrl;//最新版本的路 ...
- 聊聊HTTP gzip压缩与常见的Android网络框架
版权声明: 欢迎转载,但请保留文章原始出处 作者:GavinCT 出处:http://www.cnblogs.com/ct2011/p/5835990.html 进入主题之前,我们先来看一下客户端与服 ...
- IDEA插件清单
zookeeper插件,方便查看zk节点信息 Maven Helper,方便解决jar包冲突 Free Mybatis plugin,自动映射mapper接口到对应查询statements gener ...
- 洗礼灵魂,修炼python(31)--面向对象编程(1)—面向对象,对象,类的了解
面向对象 1.什么是面向对象 (图片来自网络) 哈哈,当然不是图中的意思. 1).面向对象(Object Oriented,OO)是软件开发方法.利用各大搜索引擎得到的解释都太官方,完全看不懂啥意思对 ...
- MySQL GTID复制错误处理之跳过错误
某Slave报错信息: mysql> show slave status\G; mysql> show slave status\G; ************************** ...
- nginx、tomcat调优方向及压测网站步骤
nginx调优方向: 1.所用事件处理模型 2.进程数 3.每个进程的连接数 4.压缩 5.缓存 tomcat调优方向: 1.内存 2.总内存 3.初始申请内存 4.线程内存 5.GC方面 tomca ...
- do-while语句及for语句(初学者)
1.do-while语句的一般形式为: do 语句 while(表达式): 这个循环与while循环的不同在于:它先执行循环中的语句,然后再判断这个表达式是否为真,如果为真则继续循环:如果为假,则中止 ...
- Android Studio 学习Demo内容及一些bug处理技巧 -----个人技术文档,两次冲刺总结
实现的基本内容 1.基本界面的注册(包括转换界面,隐式,显式注册,主界面的入口注册) 2.匿名内部类实现Button按钮的监听事件,并通过Toast进行显示 3.界面切换(显式.隐式) 4.调用浏览器 ...
- Java基本数据类型转换
一:Java的基本数据类型和引用数据类型 1:基本数据类型 2:引用数据类型 二:基本数据的类型转换 基本数据类型中,布尔类型boolean占有一个字节,由于其本身所代码的特殊含义,boolean类型 ...
- IO_ObjectOutputStream(对象的序列化)
对象序列化就是将一些对象写入到硬盘中存储起来,以便下次复用 import java.io.FileInputStream; import java.io.FileOutputStream; impor ...