2-SAT 及 一点习题
今天简单学习了一下2-SAT。现在简单地总结一下。至于定义之类的就不写了,这里就写写做法,以防以后忘记。
构图
每个值a,拆为两个点,一个表示a,一个表示^a(非a)。每个点我们可以看成是一个命题(这是我的理解)。
图中如果有一条边有X连向Y,表示如果X为真,那么可以推出Y为真。
注意,这里的图是有对称性的,这样下面的算法正确性才成立。举个例子,如果a连向b,那么必然有^b连向^a;如果^a连向b,那么必然有^b连向a。
算法
构好图后我们对其使用强连通分量算法。接着我们就可以得到一个有向无环图了,注意,这里的有向无环图依然满足对称性,这个证明比较简单。
然后我们还需要一个拓扑序,这个是强连通分量是自带的一个结果(根据染色的次序就可以判断)。
那么对于每个布尔变量x,让:
x所在的强连通分量的拓扑序在^x所在的强连通分量的拓扑序之后\(\Leftrightarrow\)x为真。
当然,如果存在x和^x在一个强连通分量里面,说明无解。否则按照上面的构造是可以得到一组解的。
正确性
一开始看到这样做,我是持有怀疑态度的,主要是下面两个情况,我想,把这些分析清楚之后,正确性就显而易见了。
情况一:x为真,y为假,但x和y是在同一个强连通分量p里面。
那么^x和^y也在同一个强连通分量q里面。
x为真,p的拓扑序在q的后面。
y为假,p的拓扑序在q的前面。
矛盾。
情况二:x为真,^x为假,y为假,^y为真,存在一条边由x连向y。
根据上面的情况一,这里我们可以把x看作x所在的强连通分量。
x为真,^x为假,x在^x的后面。
存在一条边由x连向y,y在x的后面。
y为假,^y为真,^y在y的后面。
这样我们得到了它们的拓扑序^x,x,y,^y。但是,
存在一条边由^y连向^x,^x在^y的后面。
矛盾。
字典序
利用贪心的思想,每次从最高位开始确定。
如果我们要使x为真,那么就是加一条由^x向x的边,如果行不通,就会使得x和^x在同一个强连通分量里面。所以,我们只需判断加了这条边后有没有一条从x到^x的路径,如果有就不可以让x为真。
所以我们要在一开始时处理出每两两点之间的连通性(\(O(nm)\)),逐一判断真假。使x为真,就加一条由^x向x的边,同理,使x为假就由x连向^x,然后继续维护连通性(\(O(n^2)\))。
事实上,我们并不需要继续维护图的连通性,也就是不加上那些边,不妨把这些边称为没用上的边。试想,如果在后面,需判断没有一条从x到^x的路径,如果需经过一些没用上的边,不妨设其中一条是由u连向v(这里注意u,v一对点),根据对称性可以知道有这样一条回路:x,u,v,^x。所以只需逐一能不能用上这些没用上的边(\(O(n)\))。
这样我们总的复杂度就是\(O(nm+n^2)\)。
这里可能会有些题目使用上述方法过不了,我们可以沿着上面的思路,写一个暴力,虽然最坏情况下依然是\(O(nm)\),但一些简单的剪枝可以使实际运行快得飞起来。
习题
这里有个神奇的解法,如果我们要满足一些数量关系,比如说是限制x的大小的,可以采用这样一个技巧。我们新建一些点,点i表示\(x \leq i\),那么点i的反面就是\(x > i\)。
如果x的大小范围比较大,那么就运用离散化的类似思想,这题的话,我们要选出最终答案可能取的所有值就可以了。
吐槽:我发现我越来越水了,debug了两天。。。。还有,为什么我tarjan的手写栈会慢这么多????!!
话说这题的challenge不会做,他还限制了老师的数量。
这题需要满足的一个东西是,若干的变量里,只能有一个为真或者没有真。假设在x1,x2,...,xn之间只有一个为真。我们可以这样构图:新建点y1,y2,...,yn,加这样一些边xi->yi+1,yi->^xi,当然,还有对应的反边^yi+1->^xi,xi->^yi。
2-SAT 及 一点习题的更多相关文章
- 浙大patB习题的一点总结
嘛嘛,patB的习题已经结束了,这些基本上没有啥特别难的,但还是有几道特别坑爹的题(o(╯□╰)o),还是把这些题的代码打包上传吧.
- 模拟CMOS集成电路 课后习题总结(2.1)
前几天开始自学拉扎维的模设教材,看之前浏览了EETOP论坛里面好多大神们对这本书的看法,当然也有人在抱怨,比如冒出“太科幻”.“一年才看完”之类恐怖的修饰语句,因此在开始看的时候就对此书充满了“敬畏” ...
- 《转载》PAT 习题
博客出处:http://blog.csdn.net/zhoufenqin/article/details/50497791 题目出处:https://www.patest.cn/contests/pa ...
- 《数据结构与算法分析:C语言描述_原书第二版》CH2算法分析_课后习题_部分解答
对于一个初学者来说,作者的Solutions Manual把太多的细节留给了读者,这里尽自己的努力给出部分习题的详解: 不当之处,欢迎指正. 1. 按增长率排列下列函数:N,√2,N1.5,N2,N ...
- SICP 习题 (1.7) 解题总结
SICP 习题 1.7 是对正文1.1.7节中的牛顿法求平方根的改进,改进部分是good-enough?过程. 原来的good-enough?是判断x和guess平方的差值是否小于0.001,这个过程 ...
- Python高级编程–正则表达式(习题)
原文:http://start2join.me/python-regex-answer-20141030/ ############################################## ...
- APUE习题8.7
看书的时候发现这个习题没有答案,于是就想把自己做的结果贴上来,和大家分享分享! 首先把题目贴上来吧: /*********** 8.10节中提及POSIX.1要求在调用exec时关闭打开的目录流.按下 ...
- C程序设计 习题之1-20 detab
码农一定是最需要动手实操的行业之一.有一句话叫,好记性不如烂笔头,牵强附会引申到这里,变成看书百遍,不如码字运行一遍.是不是有那么一点味道?哈哈! 这几天看的<C程序设计>,看完每个章节还 ...
- 统计学习导论:基于R应用——第三章习题
第三章习题 部分证明题未给出答案 1. 表3.4中,零假设是指三种形式的广告对TV的销量没什么影响.而电视广告和收音机广告的P值小说明,原假设是错的,也就是电视广告和收音机广告均对TV的销量有影响:报 ...
随机推荐
- QT实现拖放文件(有例子,并且图文并茂,非常清楚)
转自:http://my.oschina.net/voler/blog/345722 目录[-] 0. 源代码下载地址 1. 简单文件拖放 2. 复杂文件拖放 3. 通过按钮来完成列表数据的转移 4. ...
- Python单元测试:unittest使用简介
一.概述 本文介绍python的单元测试框架unittest,这是Python自带的标准模块unittest.unittest是基于java中的流行单元测试框架junit设计的,其功能强大且灵活,对于 ...
- Windows Azure HDInsight 现已正式发布!
今天,我们宣布正式发布 Windows Azure HDInsight 服务.HDInsight 是 Microsoft 提供的基于 Hadoop 的服务,为云提供 100% 的 Apache Had ...
- hdoj 1269 迷宫城堡(强连通分量)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1269 思路分析:该问题要求判断是否每两个房间都可以相互到达,即求该有向图中的所有点是否只构成一个强连通 ...
- poj 2593 Max Sequence(线性dp)
题目链接:http://poj.org/problem?id=2593 思路分析:该问题为求给定由N个整数组成的序列,要求确定序列A的2个不相交子段,使这m个子段的最大连续子段和的和最大. 该问题与p ...
- java面试题系列12
1.面向对象的特征有哪些方面 a.抽象: 抽象就是忽略一个主题中与当前目标无关的那些方面,以便更充分地注意与当前目标有关的方面.抽象并不打算了解全部问题,而只是选择其中的一部分,暂时不用部分细节.抽象 ...
- 裸机离奇事件:Freescale usb 有关fault
裸机离奇事件:Freescale usbucosiiFreescale\KSDK_1.2.0\examples\twrk65f180m\demo_apps\usb\host\cdc\cdc_seria ...
- 第一个processing程序(2016-01-15)
前几天下载和安装了最新的 processing,今天试一下,哈哈,真是简单之极啊,果然是给非程序员使用的,现在,我也是非程序员了.
- BZOJ 1782: [Usaco2010 Feb]slowdown 慢慢游( BIT + dfs )
orz...hzwer 对着大神的 code 看 , 稍微理解了. 考虑一只牛到达 , 那它所在子树全部 +1 , 可以用BIT维护 --------------------------------- ...
- 运行PHP
/usr/local/php/bin/php /home/www/index.php